说到数据绑定,其实这并不是一个新的玩意儿。了解asp.net的朋友都知道,在asp.net中已经用到了这个概念,例如Repeater等的数据绑定。那么,在WPF中的数据绑定相比较传统的asp.net中的数据绑定又有哪些优点呢?

1)具有双向性,即从源到目标是双向的
2)及时更新,源发生改变时,能够及时更新UI
3)Validation和Converter,前者保证数据的合法性,后者保证数据的有效性
接下来,我们将从这么几个方面来说明:Binding对象(对应xaml中的Binding扩展标记)、Binding的Path以及Source、Validation及Converter和MultiBinding。

1.Binding对象

在Binding对象中,主要成员可以分为这么几类:
1)路径:Path属性和XPath属性
2)源:Source、RelativeSource和ElementName
3)更新通知:NotifyOnSourceUpdated、NotifyOnTargetUpdated和NotifyOnValidationError
4)转换器:Converter、ConverterCulture和ConverterParameter
5)验证:ValidatesOnDataErrors、ValidatesOnExceptions、ValidatesOnNotifyDataErrors和ValidationRules
6)绑定方式:Mode,BindingMode枚举类型:TwoWay,OneWay,OneTime,OneWayToSource和Default
需要注意的是:Binding的目标必须是依赖对象的某个依赖属性。

2.Binding的Path以及Source

对于Binding的Path及Source,并非要是依赖属性及依赖对象。几乎任何一个对象都可以作为Binding的Source,主要有普通CLR对象、ado.net对象、XML、Linq、依赖对象、容器的DataContext、RelativeSource和ObjectDataProvider等。
而Path就是普通CLR对象、容器的DataContext、RelativeSource和Linq的某个属性、
ado.net对象的某个字段、依赖对象的某个依赖属性和ObjectDataProvider的某个方法名。
这里需要注意一下几点:

1)有Path没Source,将去找其父元素有该Path的DataContext;
2)有Source没Path,则将Source也作为其Path;

3)无Source无Path,则将其父元素的DataContext既作为Source也作为Path。

关于Binding的Source这里只说下Xml、Linq to Xml和ObjectDataProvider,其它几种略过。

2.1使用Xml作为Binding Source

首先,我们来准备xml数据,命名为Students.xml,如下:

<?xml version="1.0" encoding="utf-8" ?>
<Students>
<Student ID="1">
<Name>Jello</Name>
<Score>80</Score>
</Student>
<Student ID="2">
<Name>Taffy</Name>
<Score>100</Score>
</Student>
</Students>

Xaml代码如下:

<Window x:Class="DataBindingDemo.XmlWindow1"
xmlns="http://schemas.microsoft.com/winfx/2006/xaml/presentation"
xmlns:x="http://schemas.microsoft.com/winfx/2006/xaml"
Title="XmlWindow1" Height="300" Width="300">
<Grid>
<ListBox x:Name="lb">
<ListBox.ItemTemplate>
<DataTemplate>
<StackPanel Orientation="Horizontal">
<TextBlock x:Name="tbID" Text="{Binding XPath=@ID}" Width="20" Foreground="Red"/>
<TextBlock x:Name="tbName" Text="{Binding XPath=Name}" Width="40" Foreground="Green"/>
<TextBlock x:Name="tbScore" Text="{Binding XPath=Score}" />
</StackPanel>
</DataTemplate>
</ListBox.ItemTemplate>
</ListBox>
</Grid>
</Window>

cs代码如下:

    /// <summary>
/// XmlWindow1.xaml 的交互逻辑
/// </summary>
public partial class XmlWindow1 : Window
{
public XmlWindow1()
{
InitializeComponent();
/*第一种写法:
XmlDocument doc = new XmlDocument();
doc.Load(@"./Students.xml");
XmlDataProvider xdp = new XmlDataProvider();
xdp.Document = doc;
xdp.XPath = "/Students/Student";
this.lb.SetBinding(ListBox.ItemsSourceProperty, new Binding(".") { Source = xdp });
*/
//第二种写法:
XmlDataProvider xdp = new XmlDataProvider();
xdp.Source = new Uri(@"F:\dotnet\WPF\学习\WPF\Demo\WpfDemo\DataBindingDemo\Students.xml");
xdp.XPath = "/Students/Student";
this.lb.DataContext = xdp;
this.lb.SetBinding(ListBox.ItemsSourceProperty, new Binding());
}
}

第一种写法借助于XmlDataProvider的Document属性,第二种写法借助于XmlDataProvider的Source属性。

如果想直接在Xaml代码里面来直接使用XmlDataProvider来进行绑定的话,需要将Xml数据放在<x:XData>...</x:XData>标签内,代码如下:

<Window x:Class="DataBindingDemo.XmlWindow"
xmlns="http://schemas.microsoft.com/winfx/2006/xaml/presentation"
xmlns:x="http://schemas.microsoft.com/winfx/2006/xaml"
Title="XmlWindow" Height="300" Width="300">
<Window.Resources>
<XmlDataProvider x:Key="xdp" XPath="Students/Student">
<x:XData>
<Students xmlns="">
<Student ID="1">
<Name>Jello</Name>
<Score>80</Score>
</Student>
<Student ID="2">
<Name>Taffy</Name>
<Score>100</Score>
</Student>
</Students>
</x:XData>
</XmlDataProvider>
</Window.Resources>
<Grid>
<ListBox x:Name="lb" ItemsSource="{Binding Source={StaticResource xdp}}">
<ListBox.ItemTemplate>
<DataTemplate>
<StackPanel Orientation="Horizontal">
<TextBlock Text="{Binding XPath=@ID}" Width="20" Foreground="Red"/>
<TextBlock Text="{Binding XPath=Name}" Width="40" Foreground="Green"/>
<TextBlock Text="{Binding XPath=Score}" />
</StackPanel>
</DataTemplate>
</ListBox.ItemTemplate>
</ListBox>
</Grid>
</Window>

效果如下:

2.2使用Linq to Xml作为Binding Source

在使用Linq to Xml作为Binding Source之前,我们当然需要准备model,这里创建一个Student实体类:

    public class Student
{
public int ID { get; set; }
public string Name { get; set; }
public int Score { get; set; }
}

Xaml代码如下:

<Window x:Class="DataBindingDemo.XmlWindow2"
xmlns="http://schemas.microsoft.com/winfx/2006/xaml/presentation"
xmlns:x="http://schemas.microsoft.com/winfx/2006/xaml"
Title="XmlWindow2" Height="300" Width="300">
<Grid>
<ListBox x:Name="lb">
<ListBox.ItemTemplate>
<DataTemplate>
<StackPanel Orientation="Horizontal">
<TextBlock x:Name="tbID" Text="{Binding Path=ID}" Width="20" Foreground="Red"/>
<TextBlock x:Name="tbName" Text="{Binding Path=Name}" Width="40" Foreground="Green"/>
<TextBlock x:Name="tbScore" Text="{Binding Path=Score}" />
</StackPanel>
</DataTemplate>
</ListBox.ItemTemplate>
</ListBox>
</Grid>
</Window>

cs代码如下:

    /// <summary>
/// XmlWindow2.xaml 的交互逻辑
/// </summary>
public partial class XmlWindow2 : Window
{
public XmlWindow2()
{
InitializeComponent(); XDocument doc = XDocument.Load(@"F:\dotnet\WPF\学习\WPF\Demo\WpfDemo\DataBindingDemo\Students.xml");
this.lb.ItemsSource = from e in doc.Descendants("Student")
select new Student
{
ID = Int32.Parse(e.Attribute("ID").Value),
Name = e.Element("Name").Value,
Score = Int32.Parse(e.Element("Score").Value)
};
}
}

2.3使用ObjectDataProvider作为Binding Source

其他对象都是针对属性作为Path的情况,而ObjectDataProvider对象主要是针对方法。所以,我们先准备一个具有Add方法的Calculate类:

    public class Calculate
{
public double Add(string d1, string d2)
{
double i, j;
if (double.TryParse(d1, out i) && double.TryParse(d2, out j))
return i + j;
else
return ;
}
}

Xaml代码如下:

<Window x:Class="DataBindingDemo.ODPWindow1"
xmlns="http://schemas.microsoft.com/winfx/2006/xaml/presentation"
xmlns:x="http://schemas.microsoft.com/winfx/2006/xaml"
Title="ODPWindow1" Height="300" Width="300">
<Grid>
<StackPanel>
<TextBox x:Name="tb1"/>
<TextBox x:Name="tb2"/>
<TextBox x:Name="tbResult"/>
</StackPanel>
</Grid>
</Window>

cs代码如下:

    /// <summary>
/// ODPWindow1.xaml 的交互逻辑
/// </summary>
public partial class ODPWindow1 : Window
{
public ODPWindow1()
{
InitializeComponent(); ObjectDataProvider odp = new ObjectDataProvider();
odp.ObjectInstance = new Calculate();
odp.MethodName = "Add";
odp.MethodParameters.Add("");
odp.MethodParameters.Add(""); this.tb1.SetBinding(TextBox.TextProperty, new Binding("MethodParameters[0]")
{
Source = odp,
BindsDirectlyToSource = true,
UpdateSourceTrigger = UpdateSourceTrigger.PropertyChanged
});
this.tb2.SetBinding(TextBox.TextProperty, new Binding("MethodParameters[1]")
{
Source = odp,
BindsDirectlyToSource = true,
UpdateSourceTrigger = UpdateSourceTrigger.PropertyChanged
});
this.tbResult.SetBinding(TextBox.TextProperty, new Binding(".")
{
Source = odp
});
}
}

效果如下:

3.Binding的Converter

数据在Binding的Target端和Source端交换时,经常会出现类型或者格式不一致的情况,这时候,我们就可以使用Converter来处理。

WPF内置了许多的Converter,例如:

<Grid.Background>
Red
</Grid.Background>

Backgroud属性是Brush抽象类型,而我们只是用一个Red字符串赋值就能达到效果,这是内置的从String类型到SolidColorBrush类型的Converter。

我们也可以实现自己的Converter,只要实现IValueConverter接口即可。

4.Binding的Validation

数据在Binding的Target端和Source端交换时,除了经常出现类型或者格式不一致,还出现数据不合法的情况。为了避免脏数据的出现,需要在交换前进行Validate。例如:

<StackPanel>
<TextBox x:Name="tb" Text="{Binding Value,ElementName=slider,UpdateSourceTrigger=PropertyChanged}"/>
<Slider x:Name="slider" Minimum="0" Maximum="99" />
</StackPanel>

这里,我们要实现的效果是在TextBox中输入一个0到99之间的数字,Slider会划到相应位置,若输入的数字不在该范围,则TextBox提示数据不合法。

Xaml代码如下:

<StackPanel>
<TextBox x:Name="tb" />
<Slider x:Name="slider" Minimum="0" Maximum="200" />
</StackPanel>

cs代码如下:

    /// <summary>
/// ConverterWnd.xaml 的交互逻辑
/// </summary>
public partial class ConverterWnd : Window
{
public ConverterWnd()
{
InitializeComponent(); Binding binding = new Binding("Value")
{
Source = this.slider,
UpdateSourceTrigger = UpdateSourceTrigger.PropertyChanged,
};
RangeValidation rv = new RangeValidation();
rv.ValidatesOnTargetUpdated = true;//验证Source
binding.NotifyOnValidationError = true;//触发Validation.ErrorEvent
binding.ValidationRules.Add(rv);
this.tb.SetBinding(TextBox.TextProperty, binding);
this.tb.AddHandler(Validation.ErrorEvent, new RoutedEventHandler(NotifyError));
}
protected void NotifyError(object sender, RoutedEventArgs e)
{
TextBox tb = sender as TextBox;
if (tb != null)
{
if (Validation.GetErrors(tb).Count > )
tb.ToolTip = Validation.GetErrors(tb)[].ErrorContent.ToString();
}
}
}
public class RangeValidation : ValidationRule
{
public override ValidationResult Validate(object value, System.Globalization.CultureInfo cultureInfo)
{
double d;
if (double.TryParse(value.ToString(), out d))
{
if (d >= && d <= )
return new ValidationResult(true, null);
}
return new ValidationResult(false, "数据不合法");
}
}

这里将Slider的最大值设为了200,当其值大于99时,由于ValidatesOnTargetUpdate=true,所以也会路由Validation.ErrorEvent事件。

效果如下:

其实,当未添加验证时,Slider的CoerceValue会强制处理。

5.MultiBinding多路绑定

经常会遇到这样的需求,要求显示“开始日期 -- 结束日期”这样的格式,这时候比较好的做法就是使用MultiBinding,当然,你也可以重新定义一个属性。

Xaml代码如下:

<Grid>
<StackPanel>
<TextBlock Text="{Binding Name}"/>
<TextBlock>
<TextBlock.Text>
<MultiBinding StringFormat="{}{0:yyyy-MM-dd}至{1:yyyy-MM-dd}">
<Binding Path="StartDate" />
<Binding Path="EndDate" />
</MultiBinding>
</TextBlock.Text>
</TextBlock>
</StackPanel>
</Grid>

cs代码如下:

    /// <summary>
/// MultiBindingWnd.xaml 的交互逻辑
/// </summary>
public partial class MultiBindingWnd : Window
{
public MultiBindingWnd()
{
InitializeComponent(); Fruit fruit = new Fruit() { Name = "Apple", StartDate = DateTime.Today, EndDate = DateTime.Today.AddYears(1) };
this.DataContext = fruit;
}
}

这里需要注意两点:

1)MultiBinding添加Binding的顺序会影响Converter

2)MultiBinding的Converter实现的是IMultiValueConverter接口

WPF学习(8)数据绑定的更多相关文章

  1. WPF学习之数据绑定

    WPF中的数据绑定提供了很强大的功能.与普通的WinForm程序相比,其绑定功能为我们提供了很多便利,例如Binding对象的自动通知/刷新,Converter,Validation Rules,Tw ...

  2. WPF学习笔记 - 数据绑定(在代码中)

    在程序代码里,有两种设置绑定的方法,一种是调用FrameworkElement或FrameContentElement对象的SetBinding实例方法. 例如: Public MainWindow( ...

  3. WPF学习09:数据绑定之 Binding to List Data

    从WPF学习03:Element Binding我们可以实现控件属性与控件属性的绑定. 从WPF学习07:MVVM 预备知识之数据绑定 我们可以实现控件属性与自定义对象属性的绑定. 而以上两个功能在实 ...

  4. WPF学习07:MVVM 预备知识之数据绑定

    MVVM是一种模式,而WPF的数据绑定机制是一种WPF内建的功能集,两者是不相关的. 但是,借助WPF各种内建功能集,如数据绑定.命令.数据模板,我们可以高效的在WPF上实现MVVM.因此,我们需要对 ...

  5. WPF学习(8)数据绑定 https://www.cnblogs.com/jellochen/p/3541197.html

    说到数据绑定,其实这并不是一个新的玩意儿.了解asp.net的朋友都知道,在asp.net中已经用到了这个概念,例如Repeater等的数据绑定.那么,在WPF中的数据绑定相比较传统的asp.net中 ...

  6. WPF学习之路初识

    WPF学习之路初识   WPF 介绍 .NET Framework 4 .NET Framework 3.5 .NET Framework 3.0 Windows Presentation Found ...

  7. WPF学习拾遗(二)TextBlock换行

    原文:WPF学习拾遗(二)TextBlock换行 下午在帮组里的同事解决一个小问题,为了以后方便,把就把它收集一下吧. 新建一个TextBlock作为最基础的一个控件,他所携带的功能相对于其他的控件要 ...

  8. (转)WPF学习资源整理

    由于笔者正在学习WPF,所以整理出网络中部分WPF的学习资源,希望对同样在学习WPF的朋友们有所帮助. 首推刘铁猛的<深入浅出WPF>系列博文 1.深入浅出WPF(1)——什么是WPFht ...

  9. WPF学习资源整理

    WPF(WindowsPresentation Foundation)是微软推出的基于Windows Vista的用户界面框架,属于.NET Framework 3.0的一部分.它提供了统一的编程模型 ...

随机推荐

  1. Context Switch and System Call

    How many Context Switches is “normal”? This depends very much on the type of application you run. If ...

  2. css 简单 返回顶部 代码及注释说明

    1. 最简单的静态返回顶部,点击直接跳转页面顶部,常见于固定放置在页面底部返回顶部功能 方法一:用命名锚点击返回到顶部预设的id为top的元素 html代码 <a href="#top ...

  3. 动态加载资源文件(ResourceDictionary)

    原文:动态加载资源文件(ResourceDictionary) 在xaml中控件通过绑定静态资源StaticResource来获取样式Style有多种方式: 1.在项目的启动文件App中<App ...

  4. uva10635 LCS映射转LIS

    题目给定 2个序列,要我们求LCS,但是序列的长度最长是250*250, LCS的时间复杂度是O(N*N),所以无法解决 我们可以第一个序列的数字,按位置,映射为1.2.3.4.5.6.7.8.9 那 ...

  5. UVA 11427 - Expect the Expected(概率递归预期)

    UVA 11427 - Expect the Expected 题目链接 题意:玩一个游戏.赢的概率p,一个晚上能玩n盘,假设n盘都没赢到总赢的盘数比例大于等于p.以后都不再玩了,假设有到p就结束 思 ...

  6. Atitit.软件的仪表板(8)--os子系统--监控资源使用情况

    Atitit.软件的仪表板(8)--os系统--资源占用监測 CPU使用 内存使用 磁盘队列 任务管理器 网络速度 插件列表( 资源管理器插件,浏览器插件,360optim) 启动项管理  (350) ...

  7. Timus 1777. Anindilyakwa 奇怪的问题计数

    The language of Australian aborigines anindilyakwa has no numerals. No anindilyakwa can say: "I ...

  8. $(&#39;#checkbox&#39;).attr(&#39;checked&#39;); 回报checked或undefined该解决方案

    $('#checkbox').attr('checked'); 返回的是checked或者是undefined,不是原来的true和false了,有关此问题的解决方法例如以下 在JQ1.6之前的版本号 ...

  9. [转载] RaspberryPi B+ WiringPi 引脚对应图

    Pin Numbering - Raspberry Pi Model B+ Numbering Scheme Expansion Header J8 Pinout (40-pin Header) Ad ...

  10. Qt中使用的编码QTextCodec::

    //如果界面上的中文依然显示乱码,那么请将main.cpp文件中的: QTextCodec::setCodecForTr(QTextCodec::codecForLocale()); //更改为: Q ...