WPF之数据绑定总结

       WPF的数据绑定是个好东西,使原本复杂的控件赋值和动态改变简单了一些。但是入门有一点困难,所以转载此篇文章。

<!------------------------------------------------------------------------------------------------------------------------------------------------------------------------------->

        最近几天高强度开发,暴露出不少问题,WPF还达不到信手拈来的地步,好些东西还要去看看以前的项目。平时还是要多总结的,层次高了之后关注的知识点才会更深入。下面总结下WPF的绑定相关,总结之前又看了一遍深入浅出WPF,结合平时用到的得出此文(以TextBox为例,覆盖常见的需求,其他控件类似,代码下载,先看代码再看解释效果更好)。

        本文主要包含以下内容:
        1.TextBox绑定后台的值(一次绑定,类似于赋值);
        2.TextBox绑定后台的值(可通过改绑定的值自动更新值);
        3.TextBox绑定另一个控件的属性值(随时更新值);
        4.TextBox绑定另一个控件的属性值(双向更新);
        5.TextBox绑定资源的值;
        6.GridView选择一行显示其信息;
        7.其他一些注意点

       1.TextBox绑定后台的值(一次绑定,类似于赋值);

        前台设计页面:
[html]  view plain  copy
  在CODE上查看代码片 派生到我的代码片
  1. <Label>tbDataFirst:</Label>  
  2. <TextBlock Name="tbDataFirst" Width="120" TextAlignment="Center"  
  3.             Text="{Binding BindData}"></TextBlock>  
        后台代码:
[csharp]  view plain  copy
  在CODE上查看代码片 派生到我的代码片
  1. private string _BindData = string.Empty;  
  2. public string BindData  
  3. {  
  4.     get  
  5.     {  
  6.         if (_BindData.Length == 0)  
  7.             _BindData = "this is BindData";  
  8.         return _BindData;  
  9.     }  
  10.     set  
  11.     {  
  12.         _BindData = value;  
  13.     }  
  14. }  
        初始化时:tbDataFirst.DataContext = this;
        前台将绑定的逻辑固定,后台给数据源,这里后台绑定的源是当前对象,前台获得当前对象的BindData属性值;

        2.TextBox绑定后台的值(可通过改绑定的值自动更新值);

        前台设计页面:
[html]  view plain  copy
  在CODE上查看代码片 派生到我的代码片
  1. <Label>tbDataSecond:</Label>  
  2. <TextBox Name="tbDataSecond" Width="120"></TextBox>  
  3. <Button Click="Button_Click">ChangeTextInfo</Button>  
        后台代码:
[csharp]  view plain  copy
  在CODE上查看代码片 派生到我的代码片
  1. private string _TextBoxData = string.Empty;  
  2. public string TextBoxData  
  3. {  
  4.     get   
  5.     {  
  6.         if (_TextBoxData.Length == 0)  
  7.             _TextBoxData = "this is data";  
  8.         return _TextBoxData;  
  9.     }  
  10.     set  
  11.     {  
  12.         if (_TextBoxData != value)  
  13.         {  
  14.             _TextBoxData = value;  
  15.             OnPropertyChanged("TextBoxData");  
  16.         }  
  17.     }  
  18. }  
  19.   
  20. public event PropertyChangedEventHandler PropertyChanged;  
  21. public virtual void OnPropertyChanged(string propertyName)  
  22. {  
  23.     if (PropertyChanged != null)  
  24.     {  
  25.         PropertyChanged(thisnew PropertyChangedEventArgs(propertyName));  
  26.     }  
  27. }  
  28.   
  29. private void Button_Click(object sender, RoutedEventArgs e)  
  30. {  
  31.     TextBoxData = DateTime.Now.ToString("HH:mm:ss.fff");  
  32. }  
        绑定时代码:
[csharp]  view plain  copy
  在CODE上查看代码片 派生到我的代码片
  1. tbDataSecond.SetBinding(TextBox.TextProperty,   
  2.     new Binding("TextBoxData") { Source = this, Mode = BindingMode.TwoWay });  
        这里要注意跟1的区别是这里能自动更新界面的值,要实现这个功能,则源对象须继承INotifyPropertyChanged接口,同时在属性值改变时触发通知事件,即OnPropertyChanged("TextBoxData")。这样每次TextBoxData属性值变化时界面将会自动更新相应的值;

        3.TextBox绑定另一个控件的属性值(随时更新值);

        前台设计页面:
[html]  view plain  copy
  在CODE上查看代码片 派生到我的代码片
  1. <StackPanel Orientation="Horizontal" Margin="5">  
  2.     <Label>Input String:</Label>  
  3.     <TextBox Name="tbInput" Width="120"></TextBox>  
  4.     <Label>Output String:</Label>  
  5.     <TextBox Name="tbOutpnput" Width="120"   
  6.                 Text="{Binding Text, ElementName=tbInput}"></TextBox>  
  7. </StackPanel>  
        这里全部有前端实现,tbOutpnput监控tbInput控件的文本值,tbInput一有变化则tbOutpnput也会跟着变化(但本质是tbInput控件通知tbOutpnput控件的);

        4.TextBox绑定另一个控件的属性值(双向更新);

        前台设计页面:
[html]  view plain  copy
  1. <StackPanel Orientation="Horizontal" Margin="5">  
  2.     <Label>TwoWayInput String:</Label>  
  3.     <TextBox Name="tbTwoWayInput" Width="120"></TextBox>  
  4.     <Label>TwoWayOutpnput String:</Label>  
  5.     <TextBox Name="tbTwoWayOutpnput" Width="120"   
  6.                 Text="{Binding Text, ElementName=tbTwoWayInput,  
  7.                 UpdateSourceTrigger=PropertyChanged}"></TextBox>  
  8. </StackPanel>  
        这里跟3的区别就在于UpdateSourceTrigger属性值设为PropertyChanged,TextBox控件的默认值是LostFocus;

        5.TextBox绑定资源文件的值;

        前台设计页面:
[html]  view plain  copy
  在CODE上查看代码片 派生到我的代码片
  1. <Window.Resources>  
  2.     <sys:String x:Key="TestInfo">Hello World!</sys:String>  
  3. </Window.Resources>  
  4. <StackPanel Orientation="Horizontal" Margin="5">  
  5.     <Label>Read from resources:</Label>  
  6.     <Label Content="{StaticResource TestInfo}"></Label>  
  7. </StackPanel>  
        注意这里需要在前台页面的Windows命名空间加上xmlns:sys="clr-namespace:System;assembly=mscorlib";

        6.GridView选择一行显示其信息;

        这里内容主要包括:GridView和ComboBox绑定数据源(第一次加载时绑定)、TextBox和ComboBox绑定GridView的选择项的详细信息;
        前台设计页面:
[html]  view plain  copy
  1. <StackPanel Margin="5">  
  2.     <ListView x:Name="lvStudent">  
  3.         <ListView.View>  
  4.             <GridView >  
  5.                 <GridViewColumn Header="Name" Width="100"   
  6.                                 DisplayMemberBinding="{Binding Name}"></GridViewColumn>  
  7.                 <GridViewColumn Header="Level"  Width="100"   
  8.                                 DisplayMemberBinding="{Binding Level.LevelInfo}">  
  9.                 </GridViewColumn>  
  10.             </GridView>  
  11.         </ListView.View>  
  12.     </ListView>  
  13. </StackPanel>  
  14. <StackPanel Orientation="Horizontal" Margin="5">  
  15.     <Label>Name:</Label>  
  16.     <TextBox Name="tbName" Width="120"  
  17.         Text="{Binding Path=SelectedItem,  
  18.                         ElementName=lvStudent,  
  19.                         Converter={StaticResource ItemToName}}"></TextBox>              
  20. </StackPanel>  
  21. <StackPanel Orientation="Horizontal" Margin="5">  
  22.     <Label>LevelInfo</Label>  
  23.     <ComboBox Name="cbLevel" AllowDrop="False" Width="180"  
  24.                 SelectedIndex="{Binding Path=SelectedItem,ElementName=lvStudent,  
  25.                     Converter={StaticResource ItemToIndex}}"></ComboBox>  
  26. </StackPanel>  
        绑定数据源:
[csharp]  view plain  copy
  在CODE上查看代码片 派生到我的代码片
  1. private void InitDataSource()  
  2. {  
  3.     lvStudent.ItemsSource = DataProvider.StudentList;  
  4.     cbLevel.ItemsSource = DataProvider.LevelList;  
  5.     cbLevel.DisplayMemberPath = "LevelInfo";  
  6. }  
        注意这里ComboBox绑定时要设置DisplayMemberPath值;
        TextBox和ComboBox绑定GridView的选择项时,由于GridView的选择项是Object的,文本下拉框无法自动获取其数据,需要自定义转换帮助类,继承自IValueConverter,具体写法如下(SelectItemConverter为例):
[csharp]  view plain  copy
  在CODE上查看代码片 派生到我的代码片
  1. [ValueConversion(typeof(object), typeof(string))]  
  2. public class SelectItemConverter : IValueConverter  
  3. {  
  4.     public object Convert(object value, Type t, object para, CultureInfo culture)  
  5.     {  
  6.         Student data = value as Student;  
  7.         return data != null ? data.Name : string.Empty;  
  8.     }  
  9.   
  10.     public object ConvertBack(object value, Type t, object para, CultureInfo culture)  
  11.     {  
  12.         return null;  
  13.     }  
  14. }  
         这里是将选择的项先转换成Student对象然后获取其Name属性数据,注意设置ValueConversion特性,前台页面相应的引入命名空间以及标记转换类(详细请对照源码);

         7.其他一些注意点

         7.1前台和后台重复绑定时以后一次的绑定为主(刚开始开始学习时看有的人前台后台都要绑一遍,后来才知道那是重复的);
         7.2触发通知事件时注意在值变化之后触发,也就是_TextBoxData = value;OnPropertyChanged("TextBoxData");不要写倒了,刚开始学习时也遇过,写反了之后会导致第一次改变值没有反应(嘿嘿,其实当时只是依葫芦画瓢,没太理解,才会犯那些错误);
        7.3如果要绑定的集合也自动更新可以使用ObservableCollection代替List,前者实现了INotifyPropertyChanged接口,在集合变化时会自动更新界面;
        7.4上面的6其实在有数据驱动的思想,具体各位可以自行学习;
         
         至此总结完成,感觉讲的不够透彻,深入理解还需自己研究。希望能对初学者有些帮助,如果有什么错误或想法,还望不吝指教,转载请保留 原文链接
          源码下载
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值