今天我们说说依赖属性

什么是依赖属性?

当然,学术定义依旧Please Baidu:https://baike.baidu.com/item/%E4%BE%9D%E8%B5%96%E5%B1%9E%E6%80%A7/637278?fr=aladdin

那么,请问你看明白了吗?明白了就请出门左转吧,同时本人也是很佩服的,本人也是经人点拨后才了解依赖属性到底是个什么东西,他真实的定义就是:

依赖属性——一个反抗旧思想旧家庭的新时代抗争青年(说白了就是个愤青)

先讲一个故事:

贾少爷,姓贾名五代,出身贵族家庭,其父贾老爷,姓贾名四代,著名商旅,其祖父贾爷爷,姓贾名三代,文人墨客,其高祖贾高祖,姓贾名二代,官宦重臣,其先祖贾剑客,姓贾名一代,侠士风流,家中历代均有家产流传给后世。流传图谱如下:

贾一代:隐士剑谱

贾二代:丞相司印

贾三代:风雅画卷

贾四代:金银重鼎

以上四件神器流传至今,均传给了贾五代

结果贾五代是个家族异类,专心研究佛法,手里最看重的宝贝是:念珠木鱼。

这就尴尬了,作为贾少爷,只想研究佛法,那四件价值连城的宝贝根本瞧不上(给我呀……),一心想要和家庭抗争,只想拿着念珠木鱼烧香拜佛,所以这事该怎么办?

好的故事到此结束。

其实这就是依赖属性,每个父类都有其自有属性,当一个子类多级继承其父类时,那么它必然继承了其父类的所有可以继承的成员,所以就会造成最后的一个叶子子类臃肿不堪,本来不想要的属性也因为继承链而被传递给子类。上面的故事用代码解释就是这样的。

    public class JiaYidai
{
public string YinShiJianPu { get; set; }
} public class JiaErdai:JiaYidai
{
public string ChengXiangSiYin { get; set; }
} public class JiaSandai : JiaErdai
{
public string FengYaHuaJuan { get; set; }
} public class JiaSidai : JiaSandai
{
public string JinYinZhongDing { get; set; }
} public class JiaWudai : JiaSidai
{
public string NianZhuMuYu { get; set; }
}

那么当我们在调用JiaWudai时肯定会出现之前的父类的属性,如图:

那么那些不相干的属性就会同步继承而来,所以依赖属性就是为了解决这个问题而产生的。

我们来看如何来解决这个问题,既然贾五代不想继承其它的属性,那么我们就将其父类的宝贝都做成依赖属性,看看会怎么样。

首先我们先建立一个依赖属性对象:

public class DependencyProperty
{
/// <summary>
/// 属性名
/// </summary>
internal string Name;
/// <summary>
/// 属性值
/// </summary>
internal object Value; /// <summary>
/// 构造函数
/// </summary>
/// <param name="name"></param>
/// <param name="propertyName"></param>
/// <param name="ownerType"></param>
/// <param name="defaultValue"></param>
private DependencyProperty(string name, Type propertyName, Type ownerType, object defaultValue)
{
this.Name = name;
this.Value = defaultValue;
}
/// <summary>
/// 创建属性的方法
/// </summary>
/// <param name="name"></param>
/// <param name="propertyName"></param>
/// <param name="ownerType"></param>
/// <param name="defaultValue"></param>
/// <returns></returns>
public static DependencyProperty CreateInstance(string name, Type propertyName, Type ownerType, object defaultValue)
{
DependencyProperty dp = new DependencyProperty(name, propertyName, ownerType, defaultValue);
return dp;
}
}

之后我们再建立贾五代的四个父级类,代码如下

public class JiaYidai
{
public static readonly DependencyProperty InstantYidai = DependencyProperty.CreateInstance("YinShiJianPu", typeof(string), typeof(JiaYidai), "隐士剑谱");
public object GetValueYidai(DependencyProperty dp)
{
return dp.Value;
}
public void SetValueYidai(DependencyProperty dp, string value)
{
dp.Value = value;
}
public string YinShiJianPu
{
get
{
return GetValueYidai(InstantYidai).ToString();
}
set
{
SetValueYidai(InstantYidai, value);
}
}
} public class JiaErdai : JiaYidai
{ public static readonly DependencyProperty InstantErdai = DependencyProperty.CreateInstance("ChengXiangSiYin", typeof(string), typeof(JiaYidai), "丞相司印");
public object GetValueErdai(DependencyProperty dp)
{
return dp.Value;
}
public void SetValueErdai(DependencyProperty dp, string value)
{
dp.Value = value;
}
public string ChengXiangSiYin
{
get
{
return GetValueErdai(InstantErdai).ToString();
}
set
{
SetValueErdai(InstantErdai, value);
}
}
} public class JiaSandai : JiaErdai
{
public static readonly DependencyProperty InstantSandai = DependencyProperty.CreateInstance("FengYaHuaJuan", typeof(string), typeof(JiaYidai), "风雅画卷");
public object GetValueSandai(DependencyProperty dp)
{
return dp.Value;
}
public void SetValueSandai(DependencyProperty dp, string value)
{
dp.Value = value;
}
public string FengYaHuaJuan
{
get
{
return GetValueSandai(InstantSandai).ToString();
}
set
{
SetValueSandai(InstantSandai, value);
}
}
} public class JiaSidai : JiaSandai
{
public static readonly DependencyProperty InstantSidai = DependencyProperty.CreateInstance("JinYinZhongDing", typeof(string), typeof(JiaYidai), "金银重鼎");
public object GetValueSidai(DependencyProperty dp)
{
return dp.Value;
}
public void SetValueSidai(DependencyProperty dp, string value)
{
dp.Value = value;
}
public string JinYinZhongDing
{
get
{
return GetValueErdai(InstantErdai).ToString();
}
set
{
SetValueErdai(InstantErdai, value);
}
}
}

然后我们再次调用

咦,为什么没有变化呢?查看源代码就会发现我们为了调用的方便,我们在依赖属性的SetValue和GetValue方法后封装了一个public的同名属性,这个属性是可以继承的,所以贾五代依旧可以看到并使用这些属性。可以通过删除父级类中的属性进行测试,在从不再详述。当然如果真的希望各个属性间不发生任何属性继承,可以取消所有继承,让各级父类分别继承这五个依赖属性,就可以实现类构成的精简

那么这个依赖属性有什么用?让我们看另外一个例子,代码如下:

public class JiaYidai
{
private static DependencyProperty Instant = DependencyProperty.CreateInstance("YinShiJianPu", typeof(string), typeof(JiaYidai), "隐士剑谱");
public object GetValueYidai(DependencyProperty dp)
{
return dp.Value;
}
public void SetValueYidai(DependencyProperty dp, string value)
{
dp.Value = value;
}
public string YinShiJianPu
{
get
{
return GetValueYidai(Instant).ToString();
}
set
{
SetValueYidai(Instant, value);
}
}
} public class JiaErdai : JiaYidai
{ private static DependencyProperty Instant = DependencyProperty.CreateInstance("ChengXiangSiYin", typeof(string), typeof(JiaYidai), "丞相司印");
public object GetValueErdai(DependencyProperty dp)
{
return dp.Value;
}
public void SetValueErdai(DependencyProperty dp, string value)
{
dp.Value = value;
}
public string ChengXiangSiYin
{
get
{
return GetValueErdai(Instant).ToString();
}
set
{
SetValueErdai(Instant, value);
}
}
} public class JiaSandai : JiaErdai
{
private static DependencyProperty Instant = DependencyProperty.CreateInstance("FengYaHuaJuan", typeof(string), typeof(JiaYidai), "风雅画卷");
public object GetValueSandai(DependencyProperty dp)
{
return dp.Value;
}
public void SetValueSandai(DependencyProperty dp, string value)
{
dp.Value = value;
}
public string FengYaHuaJuan
{
get
{
return GetValueSandai(Instant).ToString();
}
set
{
SetValueSandai(Instant, value);
}
}
} public class JiaSidai : JiaSandai
{
private static DependencyProperty Instant = DependencyProperty.CreateInstance("JinYinZhongDing", typeof(string), typeof(JiaYidai), "金银重鼎");
public object GetValueSidai(DependencyProperty dp)
{
return dp.Value;
}
public void SetValueSidai(DependencyProperty dp, string value)
{
dp.Value = value;
}
public string JinYinZhongDing
{
get
{
return GetValueErdai(Instant).ToString();
}
set
{
SetValueErdai(Instant, value);
}
}
} public class JiaWudai : JiaSidai
{
public string NianZhuMuYu { get; set; }
}

再运行如图所示代码:

看结果:

咦,本来改的是子类,怎么把父类给改了?没错这就是依赖属性的好处,一处修改,大家都知道了,想想也知道,假如贾五代这天闲来无事,打开了风雅画卷,突然无聊,拿起笔来在画卷上写了个hello world,然后把画卷收回去了,他的祖父贾三代再拿出来打开的时候hello world会没了么?写上了不擦肯定不会消失的,这就是依赖属性的好处。

所以如果依赖属性和使用了依赖属性的对象再继承一个接口INotifyPropertyChanged的话,好像……诞生了一个新的世界,这个世界让我们下回分解……

WPF入门(2)——依赖属性的更多相关文章

  1. WPF 精修篇 依赖属性

    原文:WPF 精修篇 依赖属性 依赖属性使用场景 1. 希望可在样式中设置属性. 2. 希望属性支持数据绑定. 3. 希望可使用动态资源引用设置属性. 4. 希望从元素树中的父元素自动继承属性值. 5 ...

  2. WPF中的依赖属性

    1. WPF中的依赖属性 依赖属性是专门基于WPF创建的.在WPF库实现中,依赖属性使用普通的C#属性进行了包装,使用方法与普通的属性是相同的. 1.1 依赖属性提供的属性功能 资源 数据绑定 样式 ...

  3. WPF学习笔记——依赖属性(Dependency Property)

    1.什么是依赖属性 依赖属性是一种可以自己没有值,并且通过Binding从数据源获得值(依赖在别人身上)的属性,拥有依赖属性的对象被称为"依赖对象". 依赖项属性通过调用 Regi ...

  4. WPF教程:依赖属性

    一.什么是依赖属性 依赖属性就是一种自己可以没有值,并且可以通过绑定从其他数据源获取值.依赖属性可支持WPF中的样式设置.数据绑定.继承.动画及默认值. 将所有的属性都设置为依赖属性并不总是正确的解决 ...

  5. [No000012D]WPF(5/7)依赖属性

    介绍 WPF带来了很多传统 Windows 应用程序没有的新特性和选择.我们已经讨论了一些 WPF 的特性,是时候更进一步介绍其他特性了.当你读完这个系列之前的文章,我希望你已经或多或少地了解了 WP ...

  6. (原创)2. WPF中的依赖属性之二

    1 依赖属性 1.1 依赖属性最终值的选用 WPF属性系统对依赖属性操作的基本步骤如下: 第一,确定Base Value,对同一个属性的赋值可能发生在很多地方.还用Button的宽度来进行举例,可能在 ...

  7. 【转】【WPF】关于依赖属性的ValidateValueCallback,PropertyChangedCallback和CoerceValueCallback的执行顺序

    三个回调对应依赖属性的验证过程,改变过程和强制转换过程. class Dobj : DependencyObject { //依赖属性包装 public int MyProperty { get { ...

  8. WPF usercontrol 自定义依赖属性

    1.依赖属性不同意一般属性,一般属性主要定义在对象中,而依赖属性是存在一个特殊的依赖属性表中.2.当我们触发改变值时,需要通过SetValue这种方式进行触发. UserControl1.xaml: ...

  9. WPF 主动触发依赖属性的 PropertyChanged

    需求背景 需要显示 ViewModel 中的 Message/DpMessage,显示内容根据其某些属性来确定.代码结构抽象如下: // Model public class Message : IN ...

  10. WPF快速入门系列(2)——深入解析依赖属性

    一.引言 感觉最近都颓废了,好久没有学习写博文了,出于负罪感,今天强烈逼迫自己开始更新WPF系列.尽管最近看到一篇WPF技术是否老矣的文章,但是还是不能阻止我系统学习WPF.今天继续分享WPF中一个最 ...

随机推荐

  1. layui 数据表格按钮事件绑定和渲染

    先看效果图 使用两种渲染方法: 1.toolbar引入模板 顶部的添加和删除按钮,右侧的三个筛选,打印,导出按钮 基础参数属性:toolbar:'#demo2', //创建 删除 添加按钮模板 < ...

  2. Docker的iptables规则在iptables重启后丢失

    前因后果 1.在跳板机上使用ansible命令测试机器B时,报错如下,于是就怀疑是网络防火墙的问题 10.10.0.86 | FAILED >> { "failed": ...

  3. GitHub+jsDelivr+PicGo 打造稳定快速、高效免费图床

    标题: GitHub+jsDelivr+PicGo 打造稳定快速.高效免费图床 作者: 梦幻之心星 347369787@QQ.com 标签: [GitHub, 图床] 目录: 图床 日期: 2019- ...

  4. 04 . Nginx的Rewrite重写

    Rewrite简介 # Rewrite对应URL Rewrite,即URL重写,就是把传入web的请求重定向到其他URL的过程. # 当运维遇到要重写情况时,往往是要程序员把重写规则写好后,发给你,你 ...

  5. Java多线程通关——基础知识挑战

    等掌握了基础知识之后,才有资格说基础知识没用这样的话.否则就老老实实的开始吧.     对象的监视器 每一个Java对象都有一个监视器.并且规定,每个对象的监视器每次只能被一个线程拥有,只有拥有它的线 ...

  6. ssh-copy-id to filter out any that...ERROR: Read from socket failed: Connection reset by peer

    ssh-copy-id bigboss1 /usr/bin/ssh-copy-id: INFO: attempting to log in with the new key(s), to filter ...

  7. Java实现 LeetCode 777 在LR字符串中交换相邻字符(分析题)

    777. 在LR字符串中交换相邻字符 在一个由 'L' , 'R' 和 'X' 三个字符组成的字符串(例如"RXXLRXRXL")中进行移动操作.一次移动操作指用一个"L ...

  8. Java实现 LeetCode 707 设计链表(环形链表)

    707. 设计链表 设计链表的实现.您可以选择使用单链表或双链表.单链表中的节点应该具有两个属性:val 和 next.val 是当前节点的值,next 是指向下一个节点的指针/引用.如果要使用双向链 ...

  9. Java实现 蓝桥杯VIP 算法训练 数的划分

    [题目描述] 将整数n分成k份,且每份不能为空,任意两份不能相同(不考虑顺序). 例如:n=7,k=3,下面三种分法被认为是相同的. 1,1,5: 1,5,1: 5,1,1: 问有多少种不同的分法. ...

  10. tab-switch 样式的添加 与 tab元素样式的切换

    要点: 1.多个div类名相同情况下添加class样式 2.siblings() 方法返回被选元素的所有同级元素.DOM 树:该方法沿着 DOM 元素的同级元素向前和向后遍历. 3.利用索引,只添加当 ...