[Windbg调试WPF的依赖属性](http://blog.csdn.net/muzizongheng/article/details/46821459)中提到了wpf的DependencyObject中DependencyProperty是如何调试查看的。
从中我们看出DO(DependencyObject)与 DP(DependencyProperty)一些内部实现。
**这篇文章我们就从源码入手, 让大家了解下依赖对象中依赖属性的值的获取和赋值。**
我们先看个DP注册的例子:
~~~
public class MyStateControl : ButtonBase
{
public MyStateControl() : base() { }
public Boolean State
{
get { return (Boolean)this.GetValue(StateProperty); }
set { this.SetValue(StateProperty, value); }
}
public static readonly DependencyProperty StateProperty = DependencyProperty.Register(
"State", typeof(Boolean), typeof(MyStateControl),new PropertyMetadata(false));
}
~~~
上述Code中MyStateControl是DO,StateProperty是DP
**1.**
**当MyStateControl进行初始化, 首先会执行StateProperty, 因为它是静态字段。从而执行DependencyProperty.Register方法。**
**
**
**2.**
**这个方法内部调用了DP的构造方法, Code如下:**
// Create property
DependencyProperty dp = new DependencyProperty(name, propertyType, ownerType, defaultMetadata, validateValueCallback);
**3.**
**DP的构造方法如下:**
private DependencyProperty( string name, Type propertyType, Type ownerType, PropertyMetadata defaultMetadata, ValidateValueCallback validateValueCallback)
{
_name = name;
_propertyType = propertyType;
_ownerType = ownerType;
_defaultMetadata = defaultMetadata;
_validateValueCallback = validateValueCallback;
Flags packedData;
lock (Synchronized)
{
packedData = (Flags) GetUniqueGlobalIndex(ownerType, name);
RegisteredPropertyList.Add( this);
}
if (propertyType.IsValueType)
{
packedData |= Flags.IsValueType;
}
if (propertyType == typeof (object))
{
packedData |= Flags.IsObjectType;
}
if (typeof (Freezable).IsAssignableFrom(propertyType))
{
packedData |= Flags.IsFreezableType;
}
if (propertyType == typeof (string))
{
packedData |= Flags.IsStringType;
}
_packedData = packedData;
}
**4.**
**第3点的Code中我们可以看到packedData初始值是 (Flags) GetUniqueGlobalIndex(ownerType, name);**
**GetUniqueGlobalIndex其实是DP的私有静态变量GlobalIndexCount++得到的。**
下来这段代码可以看出:
[Flags]
private enum Flags : int
{
GlobalIndexMask = 0x0000FFFF,
IsValueType = 0x00010000,
IsFreezableType = 0x00020000,
IsStringType = 0x00040000,
IsPotentiallyInherited = 0x00080000,
IsDefaultValueChanged = 0x00100000,
IsPotentiallyUsingDefaultValueFactory = 0x00200000,
IsObjectType = 0x00400000,
// 0xFF800000 free bits
}
public int GlobalIndex
{
get { return (int) (_packedData & Flags.GlobalIndexMask); }
}
_packedData的低4位即代表了StateProperty在整个DP数组RegisteredPropertyList中的索引。
**5.**
**我们在构造里看到_packedData成员变量, 还记得我们[“windbg如何调试依赖属性”](http://blog.csdn.net/muzizongheng/article/details/46821459)用到了它吗?**
**我们用 .formats 命令转换去掉_packedData高位得到了DP在DO中的存储索引。**
**通过第4和第5点, 想必大家已经对DP注册有了了解**
**
**
**接下来我们再看下DO中如何获取DP值,以及如何设置DP值。**
**6.**
**首先我们说下DO设置DP,Code类似:**
~~~
set { this.SetValue(StateProperty, value); }
~~~
~~~
~~~
~~~
可以看到我们通过DO的SetValue来给DP设置值。
~~~
~~~
~~~
**7.**
**SetValue内部实现如下:**
private void SetValueCommon(
DependencyProperty dp,
object value,
PropertyMetadata metadata,
bool coerceWithDeferredReference,
bool coerceWithCurrentValue,
OperationType operationType,
bool isInternal)
{
。。。。。。
EntryIndex entryIndex = LookupEntry(dp.GlobalIndex);
。。。。。。
**大家可以看到DO根据DP的GlobalIndex在_effectiveValues数组中查找到EntryIndex, EntryIndex包含对应index和Value,如果没有查到则在_effectiveValues中插入并返回index。**
**(有兴趣可以看看LookupEntry的实现)**
**在数组中找到之后接下来就是往数组中赋值。代码类似(真实比下面更复杂):**
if (entryIndex.Found)
{
newEntry = _effectiveValues[entryIndex.Index];
}
**然后调用UpdateEffectiveValue发送属性更改通知。(还有其他一些Coerce相关代码,暂且不述)**
**
// fire change notification
NotifyPropertyChange(
new DependencyPropertyChangedEventArgs(
dp,
metadata,
isAValueChange,
oldEntry,
newEntry,
operationType));
通过上面我们可以了解依赖对象中的依赖属性的赋值实现, 我们接下来再看看取值。
**8.**
**DO获取DP的值,Code类似:**
~~~
~~~
~~~
get { return (Boolean)this.GetValue(StateProperty); }
~~~
**9. **
**GetValue内部实现如下:**
public object GetValue(DependencyProperty dp)
{
。。。
// Call Forwarded
return GetValueEntry(
LookupEntry(dp.GlobalIndex),
dp,
null,
RequestFlags.FullyResolved).Value;
}
可以看出是先找到DP的索引,然后接下来从_effectiveValues数组中找到对应的值。代码类似如下:
entry = _effectiveValues[entryIndex.Index];
(当然其中也有一些值优先级的处理,从来获取到正确的值)
**
**
**OK, 我们再回想下Windbg中查看依赖对象的依赖属性的值的步骤, 1.得到依赖对象的值;2.得到依赖属性的值;3得到依赖属性的GlobalIndex;4.根据GlobalIndex去依赖对象中的_effectiveValues找到对应index的值。**
**是不是对DP和DO的一些实现更了解了呢?**
~~~
~~~
- 前言
- win32与WPF的混合编程
- WPF: 一个可以用StoryBoard动态改变Grid行宽/列高的类
- MFC中调用WPF教程
- Expression Blend操作: 使用behavior来控制Storyboard
- WPF DatePicker 的textbox的焦点
- WPF 使用MultiBinding ,TwoWay ,ValidationRule ,需要注意的事项
- WPF TreeView 后台C#选中指定的Item, 需要遍历
- WPF GridViewColumn Sort DataTemplate
- DataGridColum的bug
- WPF Get Multibinding Expression, Update Source,
- WPF 后台触发 Validate UI‘s Element
- WPF ValidationRule 触发ErrorTemplate 的注意事项
- WPF DelegateCommand CanExecute
- WPF TextBox PreviewTextInput handle IME (chinese)
- No overload for 'OnStartup' matches delegate 'System.Windows.StartupEventHandler'
- WPF error: does not contain a static 'Main' method suitable for an entry point
- WPF GridView中的CellTemplate失效的原因
- DataGrid 显示选中的item
- 如何得到WPF中控件绑定的EventTrigger
- 选中DataGrid的Cell而不是row
- ContextMenu的自定义
- 输入框只能输入英文
- TextBox的OnTextboxChanged事件里对Text重新赋值带中文, 导致崩溃
- DataGrid当列宽超出当前宽度时,没有数据也恒有滚动条
- wpf如何获取control template里的元素
- Set connectionId threw an exception.
- WPF中Visible设为Collapse时,VisualTreeHelper.GetChildrenCount为0
- XAML 编码规范 (思考)
- 如何为现有控件的DependencyProperty添加Value Changed事件?
- TreeView滚动TreeViewItem
- 为BindingList添加Sort
- WPF Background的设置有坑
- 自定义Panel中添加依赖属性需要注意的问题
- TextBlock截断字符显示为....
- DataGrid 支持字符截断显示
- TreeView控件实践
- WPF如何更改系统控件的默认高亮颜色 (Highlight brush)
- ViewModel中C# Property自动添加OnPropertyChanged处理的小工具, 以及相应Python知识点
- WPF中Xaml编译正常而Designer Time时出错的解决办法
- 关于Snoop的用法
- wpf中为DataGrid添加checkbox支持多选全选
- WPF中DataGrid控件的过滤(Filter)性能分析及优化
- wpf控件提示Value ‘’ can not convert
- DropShadowEffect导致下拉框控件抖动
- 再论WPF中的UseLayoutRounding和SnapsToDevicePixels
- WPF案例:如何设计历史记录查看UI
- WPF案例:如何设计搜索框(自定义控件的原则和方法)
- WPF基本概念入门
- WPF开发中Designer和码农之间的合作
- 聊聊WPF中的Dispatcher
- 聊聊WPF中字体的设置
- Bug:DataGridCell的显示不完整
- WPF中ToolTip的自定义
- WPF中ItemsControl绑定到Google ProtocolBuffer的结构体时的性能问题
- TreeView的性能问题
- Xaml中string(字符串)常量的定义以及空格的处理
- 依赖属性
- WPF中的CheckBox的_ (underscore / 下划线)丢失
- WPF错误:必须使“Property”具有非 null 值。
- WPF中ItemsControl应用虚拟化时找到子元素的方法
- WPF毫秒级桌面时钟的实现-C#中Hook(钩子)的应用
- KB2464222导致IsNonIdempotentProperty方法找不见
- WPF中PreviewMouseDownEvent的系统处理:TabItem的PreviewMouseDown 事件弹框后不切换的问题调查
- WPF文字渲染相关的问题及解决
- wpf中的默认右键菜单中的复制、粘贴、剪贴等没有本地化的解决方案
- WPF内部DeliverEvent读锁和PrivateAddListener写锁导致死锁
- Windbg调试WPF的依赖属性
- WPF 后台Render线程崩溃, Exception from HRESULT: 0x88980406
- WPF中DependencyObject与DependencyProperty的源码简单剖析
- 禁用WPF中DataGrid默认的鼠标左键拖动多选行的效果
- wpf工程中在Xaml文件下添加多个cs文件
- ScrollViewer滚动到底来触发加载数据的Behavior