转载:如何优雅的实现INotifyPropertyChanged接口
轉(zhuǎn)載來源:http://www.cnblogs.com/TianFang/p/6240933.html
如何優(yōu)雅的實(shí)現(xiàn)INotifyPropertyChanged接口
INotifyPropertyChanged接口在WPF或WinFrom程序中使用還是經(jīng)常用到,常用于通知界面屬性變更。標(biāo)準(zhǔn)寫法如下:
????class?NotifyObject?:?INotifyPropertyChanged
????{
????????private?int?number;
????????public?int?Number
????????{
????????????get?{?return?number; }
????????????set?{ number =?value; OnPropertyChanged("Number"); }
????????}
????????private?string?text;
????????public?string?Text
????????{
????????????get?{?return?text; }
????????????set?{ text =?value; OnPropertyChanged("Text"); }
????????}
????????public?event?PropertyChangedEventHandler?PropertyChanged;
????????protected?void?OnPropertyChanged(string?propertyName =?"")
????????{
????????????PropertyChangedEventHandler?handler = PropertyChanged;
????????????if?(handler !=?null)
????????????{
????????????????handler(this,?new?PropertyChangedEventArgs(propertyName));
????????????}
????????}
????}
這種寫法的兩個(gè)問題是
?我在博客文章使用CallerMemberName簡化InotifyPropertyChanged的實(shí)現(xiàn)中介紹了通過Caller Information解決屬性名稱字符串硬編碼的問題。但是仍然不能解決冗余代碼的問題。對于第二個(gè)問題,往往是通過AOP的方式實(shí)現(xiàn),一般的實(shí)現(xiàn)方式有兩種:
我增在項(xiàng)目中使用過DynamicObject封裝來實(shí)現(xiàn)過,主要原理是用實(shí)現(xiàn)一個(gè)PocoNotifyWrapper的DynamicObject類,托管其屬性的讀寫動(dòng)作,并附加IntofyPropertyChanged接口實(shí)現(xiàn)。
這種方式是動(dòng)態(tài)的AOP了,是一個(gè)通用的方式,并且擴(kuò)展性比較器,可以通過繼承PocoNotifyWrapper來實(shí)現(xiàn)多態(tài)。用它做ViewMode層還是比較方便。
不過一個(gè)不大好的地方是DynamicObject是丟失了屬性類型信息的,有時(shí)綁定時(shí)會(huì)出錯(cuò)(如將一個(gè)TextBox的Text綁定到一個(gè)封裝后的Int類型對象時(shí)會(huì)不進(jìn)行自動(dòng)類型轉(zhuǎn)換)。要解決它還需要實(shí)現(xiàn)一些其它的接口,實(shí)現(xiàn)起來還是有些復(fù)雜的,并且項(xiàng)目進(jìn)度比較緊張,沒有太多時(shí)間來完善它。
另外一種方式是通過編譯期間代碼注入方式來實(shí)現(xiàn):
最開始見的是PostSharp的一個(gè)實(shí)現(xiàn):?http://doc.postsharp.net/inotifypropertychanged-add。不過PostSharp是收費(fèi)的,后來也逐漸由了許多其它的免費(fèi)的解決方案。本文這里介紹的是一個(gè)開源的解決方案:Fody。
使用它非常簡單,首先通過Nuget安裝庫:PM> Install-Package PropertyChanged.Fody。然后在需要實(shí)現(xiàn)屬性通知的類上加一個(gè)[ImplementPropertyChanged]即可:
]
public?class?Person
{
public?string?GivenNames?{?get;?set; }
public?string?FamilyName?{?get;?set; }
public?string?FullName?=>?string.Format("{0}?{1}",?GivenNames,?FamilyName);
}
?
編譯后生成的代碼如下:
?View Code基本上對代碼沒有注入,不需要安裝插件,也不影響調(diào)試,實(shí)現(xiàn)非常簡單,非常方便。
需要注意的是,如果實(shí)現(xiàn)了INotifyPropertyChanged接口,即使沒有[ImplementPropertyChanged]標(biāo)記,默認(rèn)也會(huì)注入。另外,如果要實(shí)現(xiàn)更多的控制,可以參看如下鏈接:
- Attributes
- BeforeAfter
- EqualityChecking
- EventInvokerSelectionInjection
- ExampleUsage
- ImplementingAnIsChangedFlag
- MVVMLightBroadcast
- NotificationInterception
- On_PropertyName_Changed
- PropertyDependencies
- SupportedToolkits
- Options
- WeavingWithoutAddingAReference
除此之外,還有許多關(guān)于實(shí)現(xiàn)的INotifyPropertyChanged文章,感興趣的朋友可以看一下:
- Aspect Examples (INotifyPropertyChanged via aspects)
- LinFu
- INotifyPropertyChanged auto wiring or how to get rid of redundant code
- INotifyPropertyChanged With Unity Interception AOP
轉(zhuǎn)載于:https://www.cnblogs.com/zhiguzhidao/p/10720794.html
總結(jié)
以上是生活随笔為你收集整理的转载:如何优雅的实现INotifyPropertyChanged接口的全部內(nèi)容,希望文章能夠幫你解決所遇到的問題。
- 上一篇: redis集群搭建及设置账户(转)
- 下一篇: Dynamics CRM中跨域调用Web