WPF依赖属性DependencyProperty
写在之前:
依赖属性算是WPF醉醉基础的一个组成了。平时写代码的时候,简单的绑定很轻松,但是遇到复杂的层次比较多的绑定,真的是要命。所以,我觉得深刻认识依赖属性是很有必要的。本篇只是个人学习的记录,学习的博客是周永恒先生的《一站式WPF--依赖属性(DependencyProperty)》,这算是一个系列了,说的很详细。如果需要更好的学习,建议移步上述原文,受益匪浅。
什么是依赖属性?
Windows Presentation Foundation (WPF) 提供了一组服务,这些服务可用于扩展公共语言运行时 (CLR) 属性的功能,这些服务通常统称为 WPF 属性系统。由 WPF 属性系统支持的属性称为依赖项属性。----MSDN
也就是说,WPF提供一组叫做‘WPF属性系统’的服务,而依赖属性就是被这个服务所支持的属性。我只能说,每个字都认识,但是放在一起认识的就不那么清晰了……
首先,想要了解它,必须知道它是为了什么而来。
虽然不清楚依赖属性,但是属性我们是很清楚的,封装类的字段,表示类的状态,编译后被转化为get_,set_方法,可以被类或结构等使用,常见的一个属性如下:
public class ClassObject
{
private string name;
public string Name
{
get { return name; }
set { name = value; }
}
}
既然已经有了属性,为什么还要有依赖属性呢?必然是属性有一些缺点了,而依赖属性恰好能够解决这个问题。以Button为例:
每次继承,父类的私有字段都被继承下来。对Button来说,大多数属性并没有被修改,仍然保持父类定义时的默认值。通常情况下,在整个Button的对象生命周期中,也只有少部分属性被修改。也已看得出来:
- 每次继承,子类对象都会获得更多的属性,这样,继承树的低端对象不可避免的膨胀;
- 既然大多数属性没有被修改,那么就可以把这些属性从对象中剥离,减少对象的体积;
可以知道,依赖属性就是为了解决这个问题诞生了。首先定义依赖属性,它里面存储之前2中希望剥离的属性:
public class DependencyProperty
{
// 维护了一个全局的Map用来储存所有的DP
internal static Dictionary<object, DependencyProperty> RegisteredDps = new Dictionary<object, DependencyProperty>();
internal string Name;//注册属性名称
internal object Value;//属性值
internal object HashCode;//Mape唯一键值 private DependencyProperty(string name, Type propertyName, Type ownerType, object defaultValue)
{
this.Name = name;
this.Value = defaultValue;
this.HashCode = name.GetHashCode() ^ ownerType.GetHashCode();
}
// 对外暴露一个Register方法用来注册新的DP
public static DependencyProperty Register(string name, Type propertyName, Type ownerType, object defaultValue)
{
DependencyProperty dp = new DependencyProperty(name, propertyName, ownerType, defaultValue);
RegisteredDps.Add(dp.HashCode, dp);
return dp;
}
}
然后定义DependencyObject来使用DP:
public class DependencyObject
{
// 注册一个新的DP:NameProperty
public static readonly DependencyProperty NameProperty = DependencyProperty.Register("Name", typeof(string), typeof(DependencyObject), string.Empty); public object GetValue(DependencyProperty dp)
{
return DependencyProperty.RegisteredDps[dp.HashCode].Value;
} public void SetValue(DependencyProperty dp, object value)
{
DependencyProperty.RegisteredDps[dp.HashCode].Value = value;
} public string Name
{
get
{
return (string)GetValue(NameProperty);
}
set
{
SetValue(NameProperty, value);
}
}
}
DependencyObject和文章开篇的ClassObject中的Name有什么不同呢?
>>DependencyObject.Name的实际值不是用字段保存在DependencyObject中,而是保存在NameProperty中,通过SetValue和GetValue来金星赋值取值操作。
在上述例子中,所有DependncuObject的对象将共用一个NameProperty,这在现实中是不实际的:只要修改一个对象的属性,相当于所有对象的属性值都被修改了。所以,修改的属性,还是要维护在相应的对象中的:(修改部分用☆表示)
public class DependencyProperty
{
private static int globalIndex = ;// ☆
// 维护了一个全局的Map用来储存所有的DP
internal static Dictionary<object, DependencyProperty> RegisteredDps = new Dictionary<object, DependencyProperty>();
internal string Name;//注册属性名称
internal object Value;//属性值
internal int Index;// ☆
internal object HashCode;//Mape唯一键值 private DependencyProperty(string name, Type propertyName, Type ownerType, object defaultValue)
{
this.Name = name;
this.Value = defaultValue;
this.HashCode = name.GetHashCode() ^ ownerType.GetHashCode();
}
// 对外暴露一个Register方法用来注册新的DP
public static DependencyProperty Register(string name, Type propertyName, Type ownerType, object defaultValue)
{
DependencyProperty dp = new DependencyProperty(name, propertyName, ownerType, defaultValue);
globalIndex++;// ☆
dp.Index = globalIndex; // ☆
RegisteredDps.Add(dp.HashCode, dp);
return dp;
}
}
所有修改过的DP都保存在EffectiveValueEntry里,这样,就可以只保存修改的属性,未修改过的属性仍然读取DP的默认值,优化了属性的储存。
public class DependencyObject
{
// 引入有效的概念// ☆
private List<EffectiveValueEntry> _effectiveValues = new List<EffectiveValueEntry>();
// 注册一个新的DP:NameProperty
public static readonly DependencyProperty NameProperty = DependencyProperty.Register("Name", typeof(string), typeof(DependencyObject), string.Empty); public object GetValue(DependencyProperty dp)
{ // ☆
EffectiveValueEntry effectiveValue = _effectiveValues.FirstOrDefault(i => i.PropertyIndex == dp.Index);
if (effectiveValue.PropertyIndex != 0)
{
return effectiveValue.Value;
}
else
{
return DependencyProperty.RegisteredDps[dp.HashCode].Value;//仅此部分相同
}
} public void SetValue(DependencyProperty dp, object value)
{ // 全部 ☆
EffectiveValueEntry effectiveValue = _effectiveValues.FirstOrDefault(i => i.PropertyIndex == dp.Index);
if (effectiveValue.PropertyIndex != 0)
{
effectiveValue.Value = value;
}
else
{
effectiveValue = new EffectiveValueEntry()
{
PropertyIndex = dp.Index,
Value = value
};
}
//DependencyProperty.RegisteredDps[dp.HashCode].Value = value;
} public string Name
{
get
{
return (string)GetValue(NameProperty);
}
set
{
SetValue(NameProperty, value);
}
}
}
internal struct EffectiveValueEntry
{
internal int PropertyIndex{get;set;}
internal object Value{get;set;}
}
WPF依赖属性DependencyProperty的更多相关文章
- WPF 依赖属性源码 洞察微软如何实现DependencyProperty
依赖属性DependencyProperty是wpf最重要的一个类,理解该类如何实现对学习wpf帮助很大! 终于找到了该类的源码!仔细阅读源码,看看微软如何玩的花招! File: Base\Syste ...
- WPF 使用依赖属性(DependencyProperty) 定义用户控件中的Image Source属性
原文:WPF 使用依赖属性(DependencyProperty) 定义用户控件中的Image Source属性 如果你要自定义一个图片按钮控件,那么如何在主窗体绑定这个控件上图片的Source呢? ...
- WPF依赖属性详解
WPF依赖属性详解 WPF 依赖属性 英文译为 Dependency Properties,是WPF引入的一种新类型的属性,在WPF中有着极为广泛的应用,在WPF中对于WPF Dependency P ...
- WPF自学入门(五)WPF依赖属性
在.NET中有事件也有属性,WPF中加入了路由事件,也加入了依赖属性.最近在写项目时还不知道WPF依赖属性是干什么用的,在使用依赖项属性的时候我都以为是在用.NET中的属性,但是确实上不是的,通过阅读 ...
- WPF依赖属性值源(BaseValueSource)
原文:WPF依赖属性值源(BaseValueSource) WPF依赖属性提供一个机制,可以获取依赖属性提供值的来源 其以BaseValueSource枚举表示 1.Default public ...
- WPF依赖属性(续)(3)依赖属性存储
原文:WPF依赖属性(续)(3)依赖属性存储 在之前的两篇,很多朋友参与了讨论,也说明各位对WPF/SL计数的热情,对DP系统各抒已见,当然也出现了一些分歧. 以下简称DP为依赖属性 ...
- WPF依赖属性(续)(2)依赖属性与附加属性的区别
原文:WPF依赖属性(续)(2)依赖属性与附加属性的区别 接上篇,感谢各位的评论,都是认为依赖属性的设计并不是为了节省内存,从大的方面而讲是如此.样式,数据绑定,动画样样都离不开它.这篇 ...
- WPF依赖属性的正确学习方法
前言 我在学习WPF的早期,对依赖属性理解一直都非常的不到位,其恶果就是,我每次在写依赖属性的时候,需要翻过去的代码来复制黏贴. 相信很多朋友有着和我相同的经历,所以这篇文章希望能帮助到那些刚刚开始学 ...
- WPF依赖属性(续)(1)
原文:WPF依赖属性(续)(1) 之前有写过几篇文章,详细地介绍了依赖属性的基本使用方法,如果你不想了解其内部实现机制的话,那么通过那两篇文章的介绍,足以应付平时的应用 ...
随机推荐
- 剑指offer:赋值运算符函数和复制构造函数
赋值运算符函数 对于定义一个赋值运算符函数时,需要注意一下几点: (1)函数的返回类型必须是一个引用,因为只有返回引用,才可以连续赋值 (2)传入的参数声明为常量引用,可以提高代码效率,同时赋值运算函 ...
- Bootstrap 模态框(Modal)插件
原文链接:http://www.runoob.com/bootstrap/bootstrap-modal-plugin.html Bootstrap 模态框(Modal)插件 模态框(Modal)是覆 ...
- Android first --- 网络编程
网络编程 ###图片下载查看 1.发送http请求 URL url = new URL(address); //获取连接对象,并没有建立连接 HttpURLConnection conn = (Htt ...
- C4.5算法的学习笔记
有日子没写博客了,这些天忙着一些杂七杂八的事情,直到某天,老师喊我好好把数据挖掘的算法搞一搞!于是便由再次埋头看起算法来!说起数据挖掘的算法,我想首先不得的不提起的就是大名鼎鼎的由决策树算法演化而来的 ...
- tsne官方论文代码解读和使用
MLGB,人生就是矫情,充满冲动,充满热恋. tsne的08年的论文看了几遍,发现原理还是蛮简单的,能想到还是不容易(人生的战场是星辰大海,但我们的贡献就是也就是宇宙尘埃) 怎么说呢,现在真的是一个好 ...
- 第一、初识C语言
1·C语言强大而灵活,如python,LISP,FORTRAN,Perl,Logo,BASIC,PASACAL的编译器和解释器都是C语言编写的. 2·C语言的指针错误往往难以察觉,但这恰好告诉我们,一 ...
- Maven打包程序
1.src/main目录下建立scritps.config.assembly目录 scritps:存放脚本文件如批处理 assembly:打包配置文件 2.assembly目录下建立package ...
- SublimeText2 快捷键一览表
SublimeText2 快捷键一览表 Sublime Text比起Notepad++,UltraEdit之类Code编辑器来说,其功能有过之而无不及,配合着各种插件使用,Sublime Text在实 ...
- .net连接DB2的异常SQL0666 - SQL query exceeds specified time limit or storage limit.错误处理
SQL0666 - SQL query exceeds specified time limit or storage limit. 原因:查询超时 解决办法: set the DbCommand.C ...
- iOS - Phone 电话
1.调用电话 1.1 拨打系统电话 调用系统自带的打电话程序,要跳转到打电话程序,打完电话自动跳转回来. 在 iOS9.0 + 系统隐私控制里禁止查询设备中已安装的 App,所以在 iOS9.0 + ...