继续聊WPF——依赖项属性(2)
在上一文中,我们用传统面向对象的方法来定义了一个类,而我们同时把该类的实例绑定到两个文本框,第一个文本框用于输入值,第二个文本框用于根据第一个文本框中的输入来取得属性值。
在上例中我们已经明了,虽然能做到同步更新,但这同步更新并不是实时的。而是在控件失去焦点或点击按钮之后才发生,因为那个时候是重新进行了绑定,所以,一般的属性声明并没有实现实时更新。
下面,我们把Student类进行改动,把Name属性改为依赖项属性。
public class Student:DependencyObject
{
//注册依赖项属性
public static readonly DependencyProperty NameProperty =
DependencyProperty.Register("Name",
typeof(string),
typeof(Student),
new PropertyMetadata(string.Empty));
}
从定义中我们看到依赖项属性的定义规则:
1、必须是公开的静态字段,public static;
2、因为是静态成员而且是公开的,有可能被恶意或无意修改,为了保险,加上一个readonly关键字;
3、调用静态方法Register返回一个DependencyProperty实例。
从上述内容中,可以进一步分析,依赖项属性是通过注册到WPF属性系统来定义的, Register方法有多种重载,示例中用到的是以下签名:
public static DependencyProperty Register(string name, Type propertyType, Type ownerType, PropertyMetadata typeMetadata);
//
// 摘要:
// 使用指定的属性名称、属性类型、所有者类型和属性元数据注册依赖项属性。
//
// 参数:
// name:
// 要注册的依赖项对象的名称。
//
// propertyType:
// 属性的类型。
//
// ownerType:
// 正注册依赖项对象的所有者类型。
//
// typeMetadata:
// 依赖项对象的属性元数据。
//
// 返回结果:
// 一个依赖项对象标识符,应使用它在您的类中设置 public static readonly 字段的值。然后,在以后使用该标识符引用依赖项对象,用于某些操作,例如以编程方式设置其值,或者获取元数据。
这里说一下ownerType参数,它只的是注册依赖项属性的类型,如本例中就是Student类。
typeMetadata是所谓的元数据,就是属性的默认,当然,它也有N个构造函数,可以同时传递事件委托来对属性的改性或类型转换时进行事件处理。
下面一点很重要,就是属性的命名格式,根据约定,必须为以下格式:
XXXProperty,你的属性名后面紧跟Property,记住!
我们把前面的示例改一下,作以下绑定
//声明一个Student类
Student myStu = new Student();
private void Window_Loaded(object sender, RoutedEventArgs e)
{
myStu.SetValue(Student.NameProperty, "小张");
//绑定到第一个文本框
BindingOperations.SetBinding(myStu, Student.NameProperty,
new Binding
{
Source = txtName,
Path = new PropertyPath("Text")
});
//绑定第二个文本框
BindingOperations.SetBinding(txtChanged, TextBox.TextProperty,
new Binding
{
Source = myStu,
Path = new PropertyPath(Student.NameProperty)
});
}
这时候你运行程序,当你在第一个文本框中输入内容时,第二个的文本框就会立即发生改变,这就说明,在双向绑定的模式下,属性的更新是实时的。
好,为了能像普通CLR属性一样使用,我们对Student类做一些封装。在封装之前,说一下是如何获取和设置属性的。
在WPF中,要使用依赖项属性,该类必须继承DependencyObject类,DependencyObject类有两个属性专门处理属性的值。
(1)GetValue用于获取值;
(2)SetValue用于设置值。
于是,我们对Student类作以下封装:
public class Student:DependencyObject
{
//注册依赖项属性
public static readonly DependencyProperty NameProperty =
DependencyProperty.Register("Name",
typeof(string),
typeof(Student),
new PropertyMetadata(string.Empty));
public string Name
{
get
{
return (string)this.GetValue(NameProperty);
}
set
{
this.SetValue(NameProperty, value);
}
}
}
虽然现在已经知道如何使用依赖项属性了,但是,依赖项属性是如何注册到WPF的属性系统中的,我们似乎一头雾水。
要把这一问题搞清楚,只有一种办法,那就是把.NET类库反编译,工具不用我说了,网上大把,呵,这可是侵权的,靠,没办法,谁叫微软不开源。
通过反编译得到Register方法的定义如下,其实每个重载都是调用了下面这个方法:
public static DependencyProperty Register(string name, Type propertyType, Type ownerType, PropertyMetadata typeMetadata, ValidateValueCallback validateValueCallback){ RegisterParameterValidation(name, propertyType, ownerType); PropertyMetadata defaultMetadata = null; if ((typeMetadata != null) && typeMetadata.DefaultValueWasSet()) { defaultMetadata = new PropertyMetadata(typeMetadata.DefaultValue); } DependencyProperty property = RegisterCommon(name, propertyType, ownerType, defaultMetadata, validateValueCallback); if (typeMetadata != null) { property.OverrideMetadata(ownerType, typeMetadata); } return property;}
其它的不用看,我们只找重点语句,上面代码片段中,加粗斜体部分为重点,也就是说,所以的依赖项属性的注册,都是调用了RegisterCommon方法的。
接着,我们来看看RegisterCommon方法是如何定义的。
这段代码有点长,我就不全部复制过来了,我把重要的语句拿过来。
private static DependencyProperty RegisterCommon(string name, Type propertyType, Type ownerType, PropertyMetadata defaultMetadata, ValidateValueCallback validateValueCallback){ FromNameKey key = new FromNameKey(name, ownerType); lock (Synchronized) { if (PropertyFromName.Contains(key)) { throw new ArgumentException(SR.Get("PropertyAlreadyRegistered", new object[] { name, ownerType.Name })); } }
//省略代码.................
return dp;}
大家注意加粗斜体字标注的地方,其中有一个if语句判断,是否在PropertyFromName中存在某个key,如果存在就抛出异常,那么这个PropertyFromName究竟是什么呢?
好,带着疑问,继续跟踪代码,在DependencyProperty类中发现了它的定义:
private static Hashtable PropertyFromName;
这下就明白了,原来刚来所检查的key的容器就是一个哈希表,而且是全局的。
上面的代码告诉我们,哈希表的key就是一个FromNameKey ,
FromNameKey key = new FromNameKey(name, ownerType);
那么,这个FromNameKey又是如何计算的呢?继续反编译,我找到了它的定义:
private class FromNameKey{ // Fields private int _hashCode; private string _name; private Type _ownerType; // Methods public FromNameKey(string name, Type ownerType) { this._name = name; this._ownerType = ownerType; this._hashCode = this._name.GetHashCode() ^ this._ownerType.GetHashCode(); } //省略代码...............} 希哈值的计算方法是:(类名 + 属性名).HashCode(); 这样一来,就可以保证依赖项属性只能注册一次,因为哈希值是全局唯一的。 而且,我们知道DependencyProperty类有个GlobalIndex,这个索引就是依赖项属性在哈希表中的位置,当我们要获取属性的值或要设属性的值, 就从这个索引表里寻找。 |
继续聊WPF——依赖项属性(2)相关推荐
- WPF学习系列之二 (依赖项属性)
依赖属性;(dependency property) 它是专门针对WPF创建的,但是WPF库中的依赖项属性都使用普通的.NET属性过程进行了包装.从而可能通过常规的方式使用它们,即使使用他们的代码 ...
- WPF中的依赖项属性
随着WPF的推广,不得不重新拾起WPF来,因为这块的产品越来越多. 只能跟着MSDN来学了,所以想是在这里记录下学习的过程和对知识的理解. 先从最基本的吧,依赖项属性是WPF中不同的地方,先了解一下它 ...
- 初步了解WPF依赖属性
一 依赖属性 在WPF库实现中,依赖属性使用普通的C#属性进行了包装,使得我们可以通过和以前一样的方式来使用依赖属性.但必须明确,在WPF中我们大多数都在使用依赖属性,而不是使用属性.依赖属性重要性在 ...
- 继续聊WPF——Slider控件
Slider控件,讲行通俗一点就是我们很常见的滑块,控件的外观上显示一系例刻度值,并存在一个可以被拖动的滑块,用户可以通过拖动滑块来控制控件的值. 如下图所示,是Windows系统的音量调节滑块. 下 ...
- slider wpf 垂直_继续聊WPF——Slider控件
下面看一个Tick控件的例子,这只是演示,Tick单独使用没有意义. Minimum="0" Fill="DarkMagenta" Placement=&quo ...
- 第四章:WPF依赖/附加属性,数据绑定(一)
总目录 前言 本章主要介绍依赖属性,附加属性以及数据绑定的的内容. 一.依赖属性&附加属性 1.依赖属性 1)概念 什么是依赖对象DependencyObject 派生自DependencyO ...
- wpf 依赖属性和附加属性
原文:wpf 依赖属性和附加属性 1.依赖属性 解释:依赖属性是配合binding出现的产物,功能主要是配合binding. 作用: 一.当自定义usercontrol时,需要与宿主(父窗体)双向绑定 ...
- VS远程开发(远程调试)编译报错:对‘xxx’未定义的引用(设置库依赖顺序)(已解决)pthread(项目-->属性-->链接器-->输入-->库依赖项)
如图在VS中对linux进行远程开发时,编译报错: 貌似是因为在代码中使用了pthread.h的函数,链接库依赖顺序出了问题,我在ubuntu里手动使用gcc main.c -lpthread -o ...
- Visual Studio 2013或2015工程属性中包含目录和库目录的添加方法,附加依赖项,相对路径
参考文章:包含目录.库目录.附加包含目录.附加库目录.附加依赖项之详解 https://blog.csdn.net/u012043391/article/details/54972127 参考文章:V ...
最新文章
- 线上内核_线上研讨会 |了解图书馆转型动态,建设智慧图书馆
- Linux TCP server系列(4)-浅谈listen与大并发TCP连接
- clion中链接openssl库
- 搭建Telnet服务器
- android关键应用程序,Android应用程序的四个关键点
- SUMIFS函数多条件求和的9个实例【转载】
- 转载:动态调用WebService(C#)
- python自定义事件event的含义_pyqt自定义事件学习出现问题
- 合并两个有序数组——C语言
- 跨站脚本攻击和跨站请求伪造
- 测试用例设计常用方法
- CDISC SDTM AE domain学习笔记 - 2
- 罗格斯的计算机科学博士奖学金,本科直博如何“牛”转乾坤斩获全奖博士录取,师兄制胜申请经历大揭秘!...
- 兼容安卓和苹果的滚动
- 要重复多少次变成潜意识_一种行为,多次重复后就能进入人的潜意识
- NW和Electron的区别
- NFS挂载失败: bad option; for several filesystems (e.g. nfs, cifs) you might need a /sbin/mount.
- rust键位失灵_switch手柄按键失灵不响应怎么办 NS手柄按键没反应解决办法
- MathPage.wll not found
- 定时监控Ubuntu系统HDMI热插拔进行锁屏操作