Binding作为数据的桥梁,连通业务逻辑层的对象(源对象)和UI的控件对象(目标对象)。在这座桥梁上,我们不仅可以控制在源对象与目标对象是双向通行还是单向通行。还可以控制数据的放行时机,甚至可以在这座桥上搭建一些关卡用来转换数据类型或者检验数据的正确性

我们先做一个最基本的例子,

创建一个"Student"类,这个类的实例将作为数据源来使用

 public class Student
{
private int _id; public int ID
{
get { return _id; }
set { _id = value; }
}
private string _name; public string Name
{
get { return _name; }
set { _name = value; }
} private int _age; public int Age
{
get { return _age; }
set { _age = value; }
} }

然后我们编写我们的控件对象和创建逻辑对象

  <StackPanel Name="stack1">
<TextBox Width="" Text="{Binding Path=ID}" HorizontalAlignment="Left" Name="txt_ID"></TextBox>
<TextBox Width="" Text="{Binding Path=Name}" HorizontalAlignment="Left" Name="txt_Name" ></TextBox>
<TextBox Text="{Binding Path=Age}" Width="" HorizontalAlignment="Left" Name="txt_Age"></TextBox>
</StackPanel>
 public Student Stu;
public MainWindow()
{
InitializeComponent();
Stu = new Student { ID = , Name = "狗娃", Age = };
       //设置元素数据绑定对象
stack1.DataContext = Stu;
}

我们可以看到TextBox控件的Text是一个{Binding} 这就是数据绑定的关键字。然后Path属性是需要绑定的属性,然后我们运行就可以看到我们已经绑定OK。

上面我们是使用的界面进行绑定,其实我们还可以使用代码对每一个控件进行绑定。现在我们将WPF界面中的绑定删除掉

 <StackPanel Name="stack1">
<TextBox Width="" HorizontalAlignment="Left" Name="txt_ID"></TextBox>
<TextBox Width="" HorizontalAlignment="Left" Name="txt_Name" ></TextBox>
<TextBox Width="" HorizontalAlignment="Left" Name="txt_Age"></TextBox>
</StackPanel>

然后我们使用代码来进行控件绑定

  public Student Stu;
public MainWindow()
{
InitializeComponent();
Stu = new Student { ID = , Name = "狗娃", Age = };
//创建Binding对象.
Binding bindingID = new Binding() {Path = new PropertyPath("ID"),Source=Stu };
//Binding构造参数可以直接传入Path参数
Binding bindingName = new Binding("Name") { Source = Stu };
Binding bindingAge = new Binding("Age") { Source = Stu }; //进行控件绑定(第一个参数是将绑定到哪个属性,第二个参数是绑定对象)
this.txt_ID.SetBinding(TextBox.TextProperty, bindingID);
this.txt_Name.SetBinding(TextBox.TextProperty, bindingName);
this.txt_Age.SetBinding(TextBox.TextProperty,bindingAge);
}

可以看到使用代码绑定需要创建Binding对象,然后使用控件的SetBinding方法进行绑定,但是郁闷的我们为了绑定这三个属性要写这么多的代码.所以使用哪种方式绑定看需求来使用.

接下来我们看双向绑定,其实上面那个我们已经实现了双向绑定,我们先做一个例子测试

创建一个测试TextBox并绑定数据ID

<TextBox Width="120"  HorizontalAlignment="Left"  Name="txt_IDTest"></TextBox>
 Binding bindingTestID = new Binding() { Path = new PropertyPath("ID"), Source = Stu };
this.txt_IDTest.SetBinding(TextBox.TextProperty, bindingTestID);

然后我们在更改txt_ID属性值后光标离开就可以看到txt_IDTest的值也随之改变了。这是因为TextBox默认是双向绑定的,所以可以改变,但是如果我们不是使用控件改变的值呢,接下来做个这样例子.在界面上添加一个Button按钮,并添加点击事件

<Button Content="Button" Width="" HorizontalAlignment="Left" Click="Button_Click"/>
 private void Button_Click(object sender, RoutedEventArgs e)
{ Stu.ID++;
}

我们在点击事件中只做了一件事,那就是让Stu的编号加1,但是运行会发现并没有改变。那么该怎么做呢。

我们需要在绑定源类型(Student类型)实现INotifyPropertyChanged接口

public class Student:INotifyPropertyChanged
{
private int _id; public int ID
{
get { return _id; }
set { _id = value; MyPropertyChanged(nameof(ID)); }
} private string _name; public string Name
{
get { return _name; }
set { _name = value; MyPropertyChanged(nameof(Name)); }
} private int _age; public event PropertyChangedEventHandler PropertyChanged; public int Age
{
get { return _age; }
set { _age = value; MyPropertyChanged(nameof(Age)); }
}
private void MyPropertyChanged(string name)
{
if(PropertyChanged!=null)
PropertyChanged(this, new PropertyChangedEventArgs(name));
}
}

然后此时我们就可以实现改变了.

2.绑定更新的计时

但是我们往往需要在输入后就让它立即改变,所以我们需要设置Binding对象中的UpdateSourceTrigger属性,

  Binding bindingID = new Binding() { Path = new PropertyPath("ID"), Source = Stu, UpdateSourceTrigger = UpdateSourceTrigger.PropertyChanged};

UpdateSourceTrigger枚举值有4个

  • Default:绑定目标属性的默认值
  • PropertyChanged:每当绑定目标属性发生改变时,都会更新数据源
  • LostFocus:每当绑定目标元素失去焦点时,都会更新绑定源
  • Explicit:仅在调用System.Windows.Data.BindingExpression.UpdateSource 方法时更新绑定源。

因此我们将UpdateSourceTrigger的属性值改成PropertyChanged即可

3.设置单项和双向绑定

刚才使用了TextBox的双向绑定,但是比如我们现在不需要双向绑定,我们只需设置Mode属性即可

 Binding bindingID = new Binding() { Path = new PropertyPath("ID"), Source = Stu,Mode = BindingMode.OneTime, UpdateSourceTrigger = UpdateSourceTrigger.PropertyChanged};

BindingMode枚举有5个值

  • TwoWay:导致更新源属性或目标属性时自动更新另一方
  • OneWay:在更改绑定源(源)时更新绑定目标(目标),如果绑定的控件为隐式只读,则此类型的绑定适用。如果无需监视目标属性的更改  则使用 System.Windows.Data.BindingMode.OneWay 绑定模式可避免 System.Windows.Data.BindingMode.TwoWay

    绑定模式的系统开销。

  • OneTime:这是实质上是 System.Windows.Data.BindingMode.OneWay 绑定的一种简化形式,它在源值不更改的情况下提供更好的性能。
  • OneWayToSource:在目标属性更改时,更新源属性。
  • Default: 使用绑定目标的默认 System.Windows.Data.Binding.Mode 值

WPF Binding学习(二)的更多相关文章

  1. WPF项目学习.二

    WPF用MVVM的解决记录 版权声明:本文为博主初学经验,未经博主允许不得转载. 一.前言 记录在学习与制作WPF过程中遇到的解决方案.  焦点的控制,键盘事件触发,输入框的数字限制,异步处理,隐藏状 ...

  2. WPF Binding学习(四) 绑定各种数据源

    转自:http://blog.csdn.net/lisenyang/article/details/18312199 1.集合作为数据源 首先我们先创建一个模型类 public class Stude ...

  3. WPF Binding学习(三)

    转自;http://blog.csdn.net/lisenyang/article/details/18312199 1.控件与控件间的双向绑定 WPF还支持控件作为数据源, <TextBox ...

  4. WPF的Binding学习笔记(二)

    原文: http://www.cnblogs.com/pasoraku/archive/2012/10/25/2738428.htmlWPF的Binding学习笔记(二) 上次学了点点Binding的 ...

  5. WPF入门教程系列(二) 深入剖析WPF Binding的使用方法

    WPF入门教程系列(二) 深入剖析WPF Binding的使用方法 同一个对象(特指System.Windows.DependencyObject的子类)的同一种属性(特指DependencyProp ...

  6. WPF Binding值转换器ValueConverter使用简介(二)-IMultiValueConverter

    注: 需要继承IMultiValueConverter接口,接口使用和IValueConverter逻辑相同. 一.MultiBinding+Converter 多值绑定及多值转换实例 当纵向流量大于 ...

  7. WPF学习二:TextBlock和Label的区别

    TextBlock和Label都是用来显示少量数据的.好多文章对Label存在的描述都是它允许使用"快速获取"."快速获取"就是允许你用Alt加上其它的按键快速 ...

  8. WPF Binding值转换器ValueConverter使用简介(一)

    WPF.Silverlight及Windows Phone程序开发中往往需要将绑定的数据进行特定转换,比如DateTime类型的时间转换为yyyyMMdd的日期,再如有一个值是根据另外多组值的不同而异 ...

  9. WPF系列教程——(二)使用Prism实现MVVM设计模式 - 简书

    原文:WPF系列教程--(二)使用Prism实现MVVM设计模式 - 简书 本文假设你已经知道MVVM设计模式是什么,所以直接进入正题,今天我们就用Prism来实现WPF的MVVM设计模式,百度上关于 ...

随机推荐

  1. jdk动态代理举例

    JDK动态代理是基于接口的代理,下面举例说明 代理类:proxy,代理动作必须要基于一个proxy实例来执行 代理执行类:实现InvocationHandler,案例中是TestInvocationH ...

  2. Django ModelForm修改默认的控件属性

    Django 中利用ModelForm 可以快速地利用数据库对应的Model 子类来自动创建对应表单. 例如: from django.db import models from django.for ...

  3. springboot 入门五-日志一

    springboot内部采用commons logging作为日志纪录,但也保留了第三方的日志框架接入的实现,例如Java Util Logging,Log4J2还有Logback.如果你要实现一种日 ...

  4. JavaScript循环实例

    几个经典的循环案例: 1.一张纸的厚度是0.0001米,将纸对折,对折多少次厚度超过珠峰高度8848米 var i=0; var h=0.0001; while(true){ i++; h=h*2; ...

  5. Python 项目实践二(生成数据)第二篇之随机漫步

    接着上节继续学习,在本节中,我们将使用Python来生成随机漫步数据,再使用matplotlib以引人瞩目的方式将这些数据呈现出来.随机漫步是这样行走得到的路径:每次行走都完全是随机的,没有明确的方向 ...

  6. CSS篇(上)

    紧接着HTML篇的CSS篇开啦,老铁们快来围观... 1.介绍一下标准的CSS盒子模型?低版本IE的盒子模型有什么不同? 1>有两种:IE盒子模型      W3C盒子模型 2>盒模型:内 ...

  7. [置顶] Xamarin android如何调用百度地图入门示例(一)

    在Xamarin android如何调用百度地图呢? 首先我们要区分清楚,百度地图这是一个广泛的概念,很多刚刚接触这个名词"百度地图api",的确是泛泛而谈,我们来看一下百度地图的 ...

  8. 在windows 10下安装python

    windows系统默认状态下是没有安装python的,我们需要下载并安装它. 首先检查是否安装了python 在"开始"菜单中输入cmd,然后右击选择管理员身份运行,这样就打开了一 ...

  9. Webpack 2 视频教程 020 - Webpack 2 中的 HMR ( Hot Module Replacement )

    原文发表于我的技术博客 这是我免费发布的高质量超清「Webpack 2 视频教程」. Webpack 作为目前前端开发必备的框架,Webpack 发布了 2.0 版本,此视频就是基于 2.0 的版本讲 ...

  10. Nifi自定义processor

    有关nifi的基本介绍和架构可以参考nifi官网 一下介绍nifi的一些比较重要的类和自己的一些理解,我刚刚接触nifi: nifi的数据流可以表示为一个flow这是一个队列,每个数据包被封装在flo ...