现在的位置: 首页 > 综合 > 正文

Windows 8开发 WinRT中MVVM的绑定属性

2013年03月24日 ⁄ 综合 ⁄ 共 1750字 ⁄ 字号 评论关闭

    对于MVVM中Model的属性,实在是开发中一个非常基础的概念,在WP7时代,我们这样写:

public class TestModel:INotifyPropertyChanged
    {
        private string _name;
        public string Name
        {
            get { return _name; }
            set
            {
                if (_name!=value)
                {
                    _name = value;
                    OnNotifyPropertyChanged("Name");
                }
            }
        }

        public event PropertyChangedEventHandler PropertyChanged;
        public void OnNotifyPropertyChanged(string propertyName)
        {
            if (PropertyChanged != null)
            {
                PropertyChanged(this, new PropertyChangedEventArgs(propertyName));
            }
        }
    }

 

  这实在是在WP7时代开发者的基本功啊。不过有没有发现,每次都需要用字符串传递的那个属性名字,硬编码,在大部分情况下,这个名字和前面的定义的属性名字是一致的,再靠手写纯属浪费(当然,很多人会偷懒一下写个代码生成器)。还有个重要的问题就是这个字符串一旦写错就悲剧了,还不容易发现。前两天看VS2012启动页上的一些视频教程有一个关于单元测试的例子,就是在说这个问题,可见这确实是个问题。

    在WIN8开发中,由于使用了.Net 4.5和C# 5.0,引入了一个新特性。在新建的Windows应用商店项目中,VS会自动生成一些代码,其中就可以发现一些端倪,不过可能有的朋友不会在第一时间去看(包括我,也是后来很偶然才发现的),所以在这里把这个发现分享给大家,先上代码:

using System.Runtime.CompilerServices;

    public class TestModel : INotifyPropertyChanged
    {
        private string _name;
        public string Name
        {
            get { return _name; }
            set { SetProperty(ref _name, value); }
        }

        public event PropertyChangedEventHandler PropertyChanged;
        protected bool SetProperty<T>(ref T storage, T value, [CallerMemberName] String propertyName = null)
        {
            if (object.Equals(storage, value)) return false;

            storage = value;
            this.OnPropertyChanged(propertyName);
            return true;
        }
        protected void OnPropertyChanged([CallerMemberName] string propertyName = null)
        {
            var eventHandler = this.PropertyChanged;
            if (eventHandler != null)
            {
                eventHandler(this, new PropertyChangedEventArgs(propertyName));
            }
        }
    }

 

 

同样的一个Name属性,用新写法之后,完成之前的功能就只需要一句话,非常简洁,而且没有硬编码。再也不用苦哈哈的手写那么复杂重复的Property属性了~

    简单说明一下,要引入System.Runtime.CompilerServices命名空间,因为SetProperty<T>需要接收的参数中的最后一项为propertyName,而这个Name我们却不想通过手工编码传递,系统提供了一个传递调用者名字的方式,就如上面代码所示,对参数加上[CallerMemberName]属性,同时把这个参数设定默认值null(这样就可以在调用时“忽略”掉这个参数,当然所有默认参数要放在参数列表的最后面)。在程序执行时,会自动传递调用者的名字,如上面是Name属性调用,传递过来的就是“Name”这个字符串。

 

 

 

锦燕云(微博:http://weibo.com/jinyanyun )

抱歉!评论已关闭.