优雅实现INotifyPropertyChanged接口——利用Lambda表达式
在14年的时候,曾经读过上面的参考文章,不过当时并没有怎么理解,慢慢地也就将这篇文章忘诸脑后了。
直到前几天突发奇想,研究如何用Lambda表达式作为构造函数的参数(详见此文),研究之后,突然联想到上文,于是自己就去把这个想法实现了一遍。
话不多说,上代码
1.基类代码 public class BaseTemplate : INotifyPropertyChanged { public event PropertyChangedEventHandler PropertyChanged; public void NotifyChanges( string propertyName) { PropertyChanged( this, new PropertyChangedEventArgs(propertyName)); } }
2.基类的扩展方法 public static class BaseTemplateExtend { public static void NotifyPropertyChanged<T, TProperty>(this T bt, Expression<Func<T, TProperty>> expression) where T: BaseTemplate { var propertyName = string.Empty; if ((expression.Body as UnaryExpression) != null) { propertyName = ((expression.Body as UnaryExpression).Operand as MemberExpression).Member.Name; } else if ((expression.Body as MemberExpression) != null) { propertyName = ((expression.Body as MemberExpression).Member.Name); } if (!string.IsNullOrEmpty(propertyName)) { bt.NotifyChanges(propertyName); } } }
3.实体类代码 public class TestTemplate : BaseTemplate { string name = “”; public string Name { get { return name; } set { name = value; this.NotifyPropertyChanges(s => s.Name); } } }
为什么要这样做? 1.微软提供的参考方案,NotifyPropertyChanges()函数里面的参数,采用的是硬编码。例如:NotifyPropertyChanges(“Name”);这样虽然很可以编译通过,但是却没办法保证我们传递参数为正确的属性名称,
而且一旦属性名称发生改变,那么我们的相应代码都需要手工来修改!这么弱智的事情,我们显然要避免。 2.基类按照我这样设置好之后,其余的实体类只需要继承基类,就可以通过示例中的方式来简单、快捷、安全地实现INotifyPropertyChanged接口。