玩转INotifyPropertyChanged和ObservableCollection

本文详细探讨了INotifyPropertyChanged接口在MVVM模式中的应用,解释了何时在实体类和ViewModel中实现该接口。此外,还分析了ObservableCollection与List的区别,包括它们在数据绑定、性能和不同场景下的适用性。最后,介绍了数据绑定的OneWay和TwoWay模式,强调了正确选择绑定方式对性能的影响。
摘要由CSDN通过智能技术生成
  本文的代码都是基于WPF的,对于Silverlight,这些技术也同样适用。


      (一)INotifyPropertyChanged的使用场合

      先写一个最简单的数据绑定,每次点击Button后,TextBlock的值都会自增1。

      效果图如下所示:

      clip_image002

      这里使用了MVVM模式,并把Click事件抽象为了Command。

      代码下载:WpfApplication4_1.zip

      观察上面的代码,注意到几个细节:

      1. UserName和Age属性作为ViewModel的两个属性,因为Age递增是基于绑定实现的,所以ViewModel要实现INotifyPropertyChanged接口。

      2. 我们只在Age上添加了OnPropertyChanged方法,它会根据Age属性的变化而自动更新XAML中绑定的值。

            而对于UserName属性,由于它是始终不变的,所以没有添加OnPropertyChanged方法。

            由此可见,OnPropertyChanged方法决定了后台数据的变化是否能影响到前台绑定的XAML。

            此外,对于一次性绑定(以后不会再改变)的属性,不要添加OnPropertyChanged方法,因为该方法会增加额外的性能开销。

      但是,在MVVM模式中,我们常常将UserName和Age属性抽象出来,作为一个实体类: 

    public class UserInfo
     {
        public string UserName { get; set; }
        public int Age { get; set; }
    }

      与此同时,在ViewModel中添加一个UserInfo类型的属性: 

    public class RegisterUserViewModel : INotifyPropertyChanged
    {
        public UserInfo User { get; set; }
    

      注意:此时XAML中的绑定相应要修改为: 

<TextBlock Name="tbUserName" Text="{Binding User.UserName}" …. />
<TextBlock Name="tbAge" Text="{Binding User. Age}" …. />

      这时候问题就来了。到底是UserInfo实现INotifyPropertyChanged接口呢,还是ViewModel实现INotifyPropertyChanged接口呢?

      1.如果UserInfo实现INotifyPropertyChanged接口,,那么UserInfo实体类相应地修改为:

复制代码
    public class UserInfo : INotifyPropertyChanged
     {
        public string UserName { get; set; }

        private int age;
        public int Age
        {
            get
            {
                return this.age;
            }
            set
            {
                if (this.age != value)
                {
                    this.age = value;
                    OnPropertyChanged("Age");
                }
            }
        }

        INotifyPropertyChanged Members
    }
复制代码

      而ViewModel相应地就简化了: 

复制代码
    public class RegisterUserViewModel
     {
        public RegisterUserViewModel()
        {
            this.User = new UserInfo { UserName = "Baobao", Age = 27 };
            ClickCommand = new DelegateCommand<object>(OnClick, arg => true);
        }

        void OnClick(object obj)
        {
            this.User.Age += 1;
        }

        public UserInfo User { get; set; }

  • 0
    点赞
  • 3
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值