彩票走势图

使用Castal DynamicProxy简化Silverlight数据绑定

转帖|其它|编辑:郝浩|2011-03-08 13:38:21.000|阅读 648 次

概述:大家都知道, 在使用 Silverlight 数据绑定的时候, 为了使源对象的更改能够传播到目标,源必须实现 INotifyPropertyChanged 接口。INotifyPropertyChanged 具有 PropertyChanged 事件,该事件通知绑定引擎源已更改,以便绑定引擎可以更新目标值。本文的内容虽然是针对 Silverlight 数据绑定而写的, 对于 WPF 数据绑定也很适用。

# 慧都年终大促·界面/图表报表/文档/IDE等千款热门软控件火热促销中 >>

  大家都知道, 在使用 Silverlight 数据绑定的时候, 为了使源对象的更改能够传播到目标,源必须实现 INotifyPropertyChanged 接口。INotifyPropertyChanged 具有 PropertyChanged 事件,该事件通知绑定引擎源已更改,以便绑定引擎可以更新目标值。

下面是一个典型的例子:

public class UserModel : INotifyPropertyChanged {

private string _firstName;
private string _lastName;

public string FirstName {
get {
return this._firstName;
}
set {
this._firstName = value;
this.NotifyPropertyChanged("FirstName");
}
}

public string LastName {
get {
return this._lastName;
}
set {
this._lastName = value;
this.NotifyPropertyChanged("LastName");
}
}

public event PropertyChangedEventHandler PropertyChanged;

protected void NotifyPropertyChanged(string propertyName) {
if (this.PropertyChanged != null) {
this.PropertyChanged(this, new PropertyChangedEventArgs(propertyName));
}
}
}

  在这个例子中,设置 FirstName 、LastName 时,需要手工激发 PropertyChanged 事件, 通知绑定引擎,因此, 如果数据源中属性比较多的时候, 是比较烦人的, 每个属性的 Setter 都需要激发一下 PropertyChanged 事件, 而且不能使用 C# 自带的自动属性特性。 当然, 可以自己设置一个代码段 snippet 来解决, 但是,重复的激发 PropertyChanged 事件的代码依然存在, 这不是我们的目标。

  前段时间看到有人在抱怨 Silverlight 的数据绑定,说必须要实现 INotifyPropertyChanged 接口, 而且还要手工调用 NotifyPropertyChanged 事件等等, 我想说的是, 借助 Castal DynamicProxy 提供的拦截技术,可以把手工调用 NotifyPropertyChanged 事件的代码省掉。

  INotifyPropertyChanged 接口是 Silverlight 数据绑定必须的, 这一点我们无法改变。因此需要先创建一个 BaseModel , 并让其实现 INotifyPropertyChanged 接口,代码如下:

public class BaseModel : INotifyPropertyChanged {

public event PropertyChangedEventHandler PropertyChanged;

public void NotifyPropertyChanged(string propertyName) {
if (this.PropertyChanged != null) {
this.PropertyChanged(this, new PropertyChangedEventArgs(propertyName));
}
}
}

  接下来为 BaseModel 写一个拦截器, 让所有继承自 BaseModel 的类在设置属性之后自动激发 NotifyPropertyChanged 事件, 拦截器代码如下:

public class NotifyPropertyChangedInterceptor : StandardInterceptor {

protected override void PostProceed(IInvocation invocation) {
base.PostProceed(invocation);
var methodName = invocation.Method.Name;
// 这里可能不是很完善, 属性的 Setter 一般都是以 set_ 开头的,
// 应该有更好的判断方法。
if (methodName.StartsWith("set_")) {
var propertyName = methodName.Substring(4);
var target = invocation.Proxy as BaseModel;
if (target != null) {
target.NotifyPropertyChanged(propertyName);
}
}
}
}

  拦截器的代码很简单, 而且是可以扩展的, 相信都能看懂, 我们还需要一个 ModelHelper , 来方便的创建 Proxy , ModelHelper 的代码如下:

public static class ModelHelper {

private static readonly ProxyGenerator ProxyGenerator = new ProxyGenerator();
private static readonly NotifyPropertyChangedInterceptor Interceptor = new NotifyPropertyChangedInterceptor();

public static T CreateProxy≶T>(T obj) where T : class, INotifyPropertyChanged {
return ProxyGenerator.CreateClassProxyWithTarget(obj, Interceptor);
}
}

  有了 ModelHelper , 可以说是万事俱备了, 我们来重写上边的 UserModel , UserModel 最终的代码如下:

public class UserModel : BaseModel {

public virtual string FirstName {
get;
set;
}

public virtual string LastName {
get;
set;
}
}

最后,使用 UserModel 的代码是这样的:

public partial class MainPage : UserControl {

public MainPage() {
InitializeComponent();

// 不能直接使用 UserModel, 要通过 ModelHelper 创建一个 Proxy 才行。
var dataContext = ModelHelper.CreateProxy(new UserModel());
dataContext.FirstName = "Zhang";
dataContext.LastName = "ZhiMin";

this.DataContext = dataContext;
}
}

  本文的内容虽然是针对 Silverlight 数据绑定而写的, 对于 WPF 数据绑定也很适用。

 

 

 

 

 

 

 


标签:

本站文章除注明转载外,均为本站原创或翻译。欢迎任何形式的转载,但请务必注明出处、不得修改原文相关链接,如果存在内容上的异议请邮件反馈至chenjj@pclwef.cn

文章转载自:网络转载

为你推荐

  • 推荐视频
  • 推荐活动
  • 推荐产品
  • 推荐文章
  • 慧都慧问
扫码咨询


添加微信 立即咨询

电话咨询

客服热线
023-68661681

TOP