Wpf基础入门——属性(依赖属性&附加属性)

本篇文章学习于: 刘铁猛老师《深入浅出WPF》

什么是属性?

属性又称为CLR属性(CLR,Common Language Runtime)。我们既可以说CLR属性是private字段的安全访问包装,也可以说一个private字段在后台支持(back)一个CLR属性。

class Person{
    private string name;

    public string Name{
        get { return name; }
        set	{ this.name=value; }
    }
}

CLR属性并不会增加内存的负担。同时也说明,属性仅仅是个语法糖(Syntax Sugar)。

依赖属性

在WPF中,微软将属性这个概念又向前推进了一步,推出了“依赖属性”这个新概念。
简言之,依赖属性就是一种可以自己没有值,并能通过使用Binding 从数据源获得值(依赖在别人身上)的属性
拥有依赖属性的对象被称为“依赖对象”。
与传统的CLR属性和面向对象思想相比依赖属性有很多新颖之处,其中包括:
(1)节省实例对内存的开销。
(2)属性值可以通过Binding依赖在其他对象上。

其实,这就是WPF中依赖属性的原理。传统的.NET开发中,一个对象所占用的内存空间在调用 new 操作符进行实例化的时候就已经决定了,而WPF允许对象在被创建的时候并不包含用于存储数据的空间(即字段所占用的空间)、只保留在需要用到数据时能够获得默认值、借用其他对象数据或实时分配空间的能力——这种对象就称为依赖对象(Dependency Object)而它这种实时获取数据的能力则依靠依赖属性(Dependency Property)来实现。WPF 开发中,必须使用依赖对象作为依赖属性的宿主,使二者结合起来,才能形成完整的 Binding目标被数据所驱动。
在WPF 系统中,依赖对象的概念被 DependencyObject类所实现,依赖属性的概念则由DependencyProperty类所实现。DependencyObject具有GetValue和 SetValue两个方法
image.png

示例:

<Window x:Class="Demo3.Wpf依赖属性.MainWindow"
  xmlns="http://schemas.microsoft.com/winfx/2006/xaml/presentation"
  xmlns:x="http://schemas.microsoft.com/winfx/2006/xaml"
  xmlns:d="http://schemas.microsoft.com/expression/blend/2008"
  xmlns:mc="http://schemas.openxmlformats.org/markup-compatibility/2006"
  xmlns:local="clr-namespace:Demo3.Wpf依赖属性"
  mc:Ignorable="d"
  Title="依赖属性" Height="450" Width="800">
  <StackPanel>
    <TextBox x:Name="tb1" TextWrapping="Wrap" Text="测试" Width="772"/>
    <TextBox x:Name="tb2" TextWrapping="Wrap" Text="TextBox" Width="772"/>
    <Button x:Name="btn1" Content="Button" Click="btn1_Click"/>
  </StackPanel>
</Window>

image.png

namespace Demo3.Wpf依赖属性 {
    /// <summary>
    /// MainWindow.xaml 的交互逻辑
    /// </summary>
    public partial class MainWindow : Window {
        public MainWindow() {
            InitializeComponent();
        }

        private void btn1_Click(object sender, RoutedEventArgs e) {
            Student stu = new Student();
            stu.SetValue(Student.NameProperty, tb1.Text);
            tb2.Text = (string)stu.GetValue(Student.NameProperty);
        }
    }

    public class Student : DependencyObject {
        public static readonly DependencyProperty NameProperty = 
            DependencyProperty.Register("Name", typeof(string), typeof(Student));
    }
}

附加属性(Attached Properties)

顾名思义,附加属性是说一个属性本来不属于某个对象,但由于某种需求而被后来附加上。也就是把对象放入一个特定环境后对象才具有的属性(表现出来就是被环培赋予的属性)就称为附加属性((Attached Properties)。
例如:放在Grid里就让Grid为它附加上 Column和Row属性,放在Canvas里就让Canvas为它附加上Top、Left等属性,放在 DockPanel里就让DockPanel 为它附加Dock属性。可见,附加属性的作用就是将属性与数据类型(宿主)解耦,让数据类型的设计更加灵活。
理解了附加属性的含义,我们开始研究附加属性的声明、注册和使用。附加属性的本质就是依赖属性,二者仅在注册和包装器上有一点区别。
可以很明显地看出,GradeProperty就是一个 DependencyProperty类型成员变量,声明时一样使用public static readonly 三个关键字共同修饰。唯一的不同就是注册附加属性使用的是名为RegisterAttached 的方法,但参数却与使用 Register方法无异。附加属性的包装器也与依赖属性不同——依赖属性使用CLR属性对GetValue和 SetValue两个方法进行包装,附加属性则使用两个方法分别进行包装——这样做完全是为了在使用的时候保持语句行文上的通畅。

示例:

<Window x:Class="Demo3.Wpf依赖属性.MainWindow"
  xmlns="http://schemas.microsoft.com/winfx/2006/xaml/presentation"
  xmlns:x="http://schemas.microsoft.com/winfx/2006/xaml"
  xmlns:d="http://schemas.microsoft.com/expression/blend/2008"
  xmlns:mc="http://schemas.openxmlformats.org/markup-compatibility/2006"
  xmlns:local="clr-namespace:Demo3.Wpf依赖属性"
  mc:Ignorable="d"
  Title="依赖属性" Height="450" Width="800">
  <StackPanel>
    <TextBox x:Name="tb1" TextWrapping="Wrap" Text="测试" Width="772"/>
    <TextBox x:Name="tb2" TextWrapping="Wrap" Text="TextBox" Width="772"/>
    <Button x:Name="btn1" Content="测试依赖属性" Click="btn1_Click"/>
    <Button x:Name="btn2" Content="测试附加属性" Click="btn2_Click"/>
  </StackPanel>
</Window>

image.png

namespace Demo3.Wpf依赖属性 {
    /// <summary>
    /// MainWindow.xaml 的交互逻辑
    /// </summary>
    public partial class MainWindow : Window {
        public MainWindow() {
            InitializeComponent();
        }

        /// <summary>
        /// 测试依赖属性
        /// </summary>
        /// <param name="sender"></param>
        /// <param name="e"></param>
        private void btn1_Click(object sender, RoutedEventArgs e) {
            Person person = new Person();
            person.SetValue(Person.NameProperty, tb1.Text);
            tb2.Text = (string)person.GetValue(Person.NameProperty);
        }

        /// <summary>
        /// 测试附加属性
        /// </summary>
        /// <param name="sender"></param>
        /// <param name="e"></param>
        private void btn2_Click(object sender, RoutedEventArgs e) {
            Person person = new Person();
            School.SetStudenID(person, 20);
            int id = School.GetStudenID(person);
            MessageBox.Show(id.ToString());
        }
    }

    public class Person : DependencyObject {
        public static readonly DependencyProperty NameProperty = DependencyProperty.Register("Name", typeof(string), typeof(Person));
    }

    public class School : DependencyObject {

        public static int GetStudenID(DependencyObject obj) {
            return (int)obj.GetValue(StudenIDProperty);
        }

        public static void SetStudenID(DependencyObject obj, int value) {
            obj.SetValue(StudenIDProperty, value);
        }

        // Using a DependencyProperty as the backing store for StudenID.  This enables animation, styling, binding, etc...
        public static readonly DependencyProperty StudenIDProperty =
            DependencyProperty.RegisterAttached("StudenID", typeof(int), typeof(School), new PropertyMetadata(0));

    }
}
posted @ 2023-05-29 21:12  不爱菠萝的菠萝君  阅读(969)  评论(0)    收藏  举报