WPF是分离UI和Logic的最佳工具,不同于Window Form的事件驱动原理,WPF采用的是数据驱动,让UI成为了Logic的附属,达到分离的效果。

本篇主要讲讲wpf的精华:data binding

可以把binding看做数据的桥梁,两端分别是source和target,一般来说,source是logic的对象,target是UI的控件对象。

(一)简单的例子

在一个界面里有一个textbox和button,按button后textbox会不停地在Student类的实例的Name属性上增加“Name”,而TextBox显示该实例的Name属性

Student类:

 using System;
 using System.Collections.Generic;
 using System.Linq;
 using System.Text;
 using System.Threading.Tasks;
 using System.ComponentModel;

 namespace WpfApplication1
 {
     class Student : INotifyPropertyChanged
     {
         public event PropertyChangedEventHandler PropertyChanged;

         private string name;

         public string Name
         {
             get { return name; }
             set {
                 name = value;
                 if (this.PropertyChanged != null)
                 {
                     this.PropertyChanged.Invoke(this, new PropertyChangedEventArgs("Name"));
                 }
             }
         }
     }
 }

MainWindow.xmal:

 <Window x:Class="WpfApplication1.MainWindow"
         xmlns="http://schemas.microsoft.com/winfx/2006/xaml/presentation"
         xmlns:x="http://schemas.microsoft.com/winfx/2006/xaml"
         Title=">
     <StackPanel>
         <TextBox x:Name="/>
         <Button Content=" Click="Button_Click"/>
     </StackPanel>
 </Window>

MainWindow.xmal.cs:

 using System;
 using System.Collections.Generic;
 using System.Linq;
 using System.Text;
 using System.Threading.Tasks;
 using System.Windows;
 using System.Windows.Controls;
 using System.Windows.Data;
 using System.Windows.Documents;
 using System.Windows.Input;
 using System.Windows.Media;
 using System.Windows.Media.Imaging;
 using System.Windows.Navigation;
 using System.Windows.Shapes;

 namespace WpfApplication1
 {
     /// <summary>
     /// Interaction logic for MainWindow.xaml
     /// </summary>
     public partial class MainWindow : Window
     {
         Student stu;
         public MainWindow()
         {
             InitializeComponent();

             stu = new Student();
             Binding binding = new Binding();
             binding.Source = stu;
             binding.Path = new PropertyPath("Name");

             BindingOperations.SetBinding(this.textBoxName, TextBox.TextProperty, binding);
         }

         private void Button_Click(object sender, RoutedEventArgs e)
         {
             stu.Name += "Name";
         }
     }
 }

可以看到在对Student类的Name属性加上PropertyChanged.Invoke函数后就能实现该属性的被绑定了。

MainWindow.xmal只是一个UI,并没有什么Logic部分

而在MainWindow.xmal.cs中,将textBoxName的TextProperty和stu的Name属性进行了绑定,Button_Click事件则只是单单地为Logic了,没有UI的代码,彻彻底底地将UI上数据变化的工作交给了binding自动处理。

从这里可以看出,WPF真的是一个非常牛叉的技术,将UI和Logic完美分离,让UI设计者和业务逻辑者可以并行地开发项目,减少了项目开发时间

当然上面这段代码还可以进行优化

MainWindow.xmal.cs:

 using System;
 using System.Collections.Generic;
 using System.Linq;
 using System.Text;
 using System.Threading.Tasks;
 using System.Windows;
 using System.Windows.Controls;
 using System.Windows.Data;
 using System.Windows.Documents;
 using System.Windows.Input;
 using System.Windows.Media;
 using System.Windows.Media.Imaging;
 using System.Windows.Navigation;
 using System.Windows.Shapes;

 namespace WpfApplication1
 {
     /// <summary>
     /// Interaction logic for MainWindow.xaml
     /// </summary>
     public partial class MainWindow : Window
     {
         Student stu;
         public MainWindow()
         {
             InitializeComponent();

             this.textBoxName.SetBinding(TextBox.TextProperty, new Binding("Name") { Source = stu = new Student() });
         }

         private void Button_Click(object sender, RoutedEventArgs e)
         {
             stu.Name += "Name";
         }
     }
 }

(二)控件作为binding source

上面的例子让一个类的属性作为source,通过让类实现INotifyPropertyChanged接口,并在属性的set语句中激发PropertyChanged事件。

除了以类的属性为source外,还有其他的选择,这里先讲第一种:控件作为源

这个时候可以将binding语句放在xaml文件中

 <Window x:Class="WpfApplication1.MainWindow"
         xmlns="http://schemas.microsoft.com/winfx/2006/xaml/presentation"
         xmlns:x="http://schemas.microsoft.com/winfx/2006/xaml"
         Title="Simple Binding" Height="110" Width="300">
     <StackPanel x:Name="stackPanel" Background="LightBlue">
         <TextBox x:Name="textBox1" Text="{Binding Path=Value, ElementName=slider1}" BorderBrush="Black" Margin="5" />
         <Slider x:Name="slider1" Maximum="100" Minimum="0" Margin="5" />
     </StackPanel>
 </Window>

这里Text="{Binding Path=Value, ElementName=slider1}"也可以在cs文件里进行绑定

 this.textBox1.SetBinding(TextBox.TextProperty, new Binding("Value") { ElementName = "slider1" });

推荐在xaml里进行绑定,控件作为source它不参与logic部分,因此没必要放入cs文件中

为了能够在textBox1里修改数据后slider1能实时更新(默认是要LostFocus才更新),可以这么写

 <TextBox x:Name="textBox1" Text="{Binding Path=Value, ElementName=slider1, UpdateSourceTrigger=PropertyChanged}" BorderBrush="Black" Margin="5" />

Binding的Path其实就是要绑定的source的具体的属性,这里是slider1的Value。如果要显示一个TextBox的文本长度,则Path=Text.Length,如果是TextBox第4个字符,Path=Text[3].

如果Path="."说明source本身就是数据,xaml里可以省略,cs里不行

(三)没有source的binding--使用DataContext作为Binding的源

有的时候binding的path指定了,source并没有指定,则UI元素树会一层一层往上爬,直到有Path指定的属性为止,如果都没有,则什么都没了

为了在xaml能够访问到cs里写的类,需要加上:xmlns:local="clr-namespace:WpfApplication1"。

先在cs里设计student类

 using System;
 using System.Collections.Generic;
 using System.Linq;
 using System.Text;
 using System.Threading.Tasks;
 using System.ComponentModel;

 namespace WpfApplication1
 {
     class Student
     {
         public int Id { get; set; }
         public string Name { get; set; }
         public int Age { get; set; }
     }
 }

再在xaml里创建UI

 <Window x:Class="WpfApplication1.MainWindow"
         xmlns="http://schemas.microsoft.com/winfx/2006/xaml/presentation"
         xmlns:x="http://schemas.microsoft.com/winfx/2006/xaml"
         xmlns:local="clr-namespace:WpfApplication1"
         Title="Simple Binding" Height="150" Width="300">
     <StackPanel x:Name="stackPanel" Background="LightBlue">
         <StackPanel.DataContext>
             <local:Student Id="6" Age="29" Name="Tim" />
         </StackPanel.DataContext>
         <Grid>
             <StackPanel>
                 <TextBox Text="{Binding Path=Id}" Margin="5" />
                 <TextBox Text="{Binding Path=Name}" Margin="5" />
                 <TextBox Text="{Binding Path=Age}" Margin="5" />
             </StackPanel>
         </Grid>
     </StackPanel>
 </Window>

如果DataContext是<sys:String>Hello DataContext!</sys:String>这种string时,Path=.,也可以将其省略了。

那么问题来了:啥时候用DataContext? 在这个例子中与其这么binding,还不如(一)里面写的改写Student类,而在xaml里进行简单的binding来得简单,而且这里xaml里需要访问cs的类,UI需要知道logic的类设计,对UI设计者来说也增加了负担,似乎不太合理。DataContext在下面两种情况下回用到

1. 当UI上的多个控件都使用Binding关注同一个对象时,不妨使用DataContext。

2. 当作为Source的对象不能被直接访问的时候,比如B窗体内的控件想把A窗体内的控件当做自己的Binding源时,但A窗体内的控件是private访问级别,这时候就可以把这个控件(或者控件的值)作为窗体A的DataContext(这个属性是public访问级别的)从而暴露数据。

形象地说,这时候外层容器的DataContext就相当于一个数据的“制高点”,只要把数据放上去,别的元素就能看见。另外,DataContext本身也是一个依赖属性,我们可以使用Binding把它关联到一个数据源上。

(四)集合对象为source

还是用上面的Student.cs,xaml为:

 <Window x:Class="WpfApplication1.MainWindow"
         xmlns="http://schemas.microsoft.com/winfx/2006/xaml/presentation"
         xmlns:x="http://schemas.microsoft.com/winfx/2006/xaml"
         xmlns:local="clr-namespace:WpfApplication1"
         Title="Simple Binding" Height="240" Width="360">
     <StackPanel x:Name="stackPanel" Background="LightBlue">
         <TextBlock Text="Student ID:" FontWeight="Bold" Margin="5" />
         <TextBox x:Name="textBoxId" Margin="5" />
         <TextBlock Text="Student List:" FontWeight="Bold" Margin="5" />
         <ListBox x:Name="listBoxStudents" Height="110" Margin="5" />
     </StackPanel>
 </Window>

xaml.cs:

 using System;
 using System.Collections.Generic;
 using System.Linq;
 using System.Text;
 using System.Threading.Tasks;
 using System.Windows;
 using System.Windows.Controls;
 using System.Windows.Data;
 using System.Windows.Documents;
 using System.Windows.Input;
 using System.Windows.Media;
 using System.Windows.Media.Imaging;
 using System.Windows.Navigation;
 using System.Windows.Shapes;

 namespace WpfApplication1
 {
     /// <summary>
     /// Interaction logic for MainWindow.xaml
     /// </summary>
     public partial class MainWindow : Window
     {
         public MainWindow()
         {
             InitializeComponent();

             List<Student> stuList = new List<Student>()
             {
                 , Name=},
                 , Name=},
                 , Name=},
                 , Name=},
                 , Name=},
                 , Name=},
             };

             this.listBoxStudents.ItemsSource = stuList;
             this.listBoxStudents.DisplayMemberPath = "Name";
             this.textBoxId.SetBinding(TextBox.TextProperty, new Binding("SelectedItem.Id") { Source = this.listBoxStudents });
         }
     }
 }

如果想要在listBox里显示Id, Name, Age,则要把this.listBoxStudents.DisplayMemberPath = "Name";得删掉,另外在xaml的listBox里设置它的ItemTemplate为特定的DataTemplate,具体如下

 <Window x:Class="WpfApplication1.MainWindow"
         xmlns="http://schemas.microsoft.com/winfx/2006/xaml/presentation"
         xmlns:x="http://schemas.microsoft.com/winfx/2006/xaml"
         Title="Simple Binding" Height="240" Width="360">
     <StackPanel x:Name="stackPanel" Background="LightBlue">
         <TextBlock Text="Student ID:" FontWeight="Bold" Margin="5" />
         <TextBox x:Name="textBoxId" Margin="5" />
         <TextBlock Text="Student List:" FontWeight="Bold" Margin="5" />
         <ListBox x:Name="listBoxStudents" Height="110" Margin="5">
             <ListBox.ItemTemplate>
                 <DataTemplate>
                     <StackPanel Orientation="Horizontal">
                         <TextBlock Text="{Binding Path=Id}" Width="30"/>
                         <TextBlock Text="{Binding Path=Name}" Width="60"/>
                         <TextBlock Text="{Binding Path=Age}" Width="30"/>
                     </StackPanel>
                 </DataTemplate>
             </ListBox.ItemTemplate>
         </ListBox>
     </StackPanel>
 </Window>

(五)ADO.NET对象为source

可以用listBox控件来显示

xaml:

 <Window x:Class="WpfApplication1.MainWindow"
         xmlns="http://schemas.microsoft.com/winfx/2006/xaml/presentation"
         xmlns:x="http://schemas.microsoft.com/winfx/2006/xaml"
         Title="Simple Binding" Height="206" Width="250">
     <StackPanel x:Name="stackPanel" Background="LightBlue">
         <ListBox x:Name="listBoxStudents" Height="130" Margin="5" />
         <Button Content="Load" Height="25" Margin="5" Click="Button_Click" />
     </StackPanel>
 </Window>

xaml.cs:

 using System;
 using System.Collections.Generic;
 using System.Linq;
 using System.Text;
 using System.Threading.Tasks;
 using System.Windows;
 using System.Windows.Controls;
 using System.Windows.Data;
 using System.Windows.Documents;
 using System.Windows.Input;
 using System.Windows.Media;
 using System.Windows.Media.Imaging;
 using System.Windows.Navigation;
 using System.Windows.Shapes;
 using System.Data;
 using MySql.Data;
 using MySql.Data.Entity;
 using MySql.Data.MySqlClient;

 namespace WpfApplication1
 {
     /// <summary>
     /// Interaction logic for MainWindow.xaml
     /// </summary>
     public partial class MainWindow : Window
     {
         public MainWindow()
         {
             InitializeComponent();
         }

         private void Button_Click(object sender, RoutedEventArgs e)
         {
             DataTable dt = this.Load();
             this.listBoxStudents.DisplayMemberPath = "Name";
             this.listBoxStudents.ItemsSource = dt.DefaultView;
         }

         private DataTable Load()
         {
             DataTable dt = new DataTable();
             using (MySqlConnection con = new MySqlConnection("server=localhost; database=persons; uid=root; pwd=0000; connect timeout=30; pooling=true"))
             {
                 con.Open();
                 MySqlDataAdapter adapter = new MySqlDataAdapter("SELECT * FROM Student", con);
                 DataSet ds = new DataSet();
                 adapter.Fill(ds, "Student");
                 dt = ds.Tables["Student"];
             }
             return dt;
         }
     }
 }

但是这样显示的内容太少了,多数情况会选择ListView来显示

xaml:

 <Window x:Class="WpfApplication1.MainWindow"
         xmlns="http://schemas.microsoft.com/winfx/2006/xaml/presentation"
         xmlns:x="http://schemas.microsoft.com/winfx/2006/xaml"
         Title="Simple Binding" Height="206" Width="250">
     <StackPanel x:Name="stackPanel" Background="LightBlue">
         <ListView x:Name="listViewStudents" Height="130" Margin="5">
             <ListView.View>
                 <GridView>
                     <GridViewColumn Header="Id" Width="60" DisplayMemberBinding="{Binding Id}" />
                     <GridViewColumn Header="Name" Width="80" DisplayMemberBinding="{Binding Name}" />
                     <GridViewColumn Header="Age" Width="60" DisplayMemberBinding="{Binding Age}" />
                 </GridView>
             </ListView.View>
         </ListView>
         <Button Content="Load" Height="25" Margin="5" Click="Button_Click" />
     </StackPanel>
 </Window>

在cs文件里注意将this.listBoxStudents.DisplayMemberPath = "Name";删除,this.listBoxStudents.ItemsSource = dt.DefaultView;改为this.listViewStudents.ItemsSource = dt.DefaultView;

(六)XML为source

先写一个Student.xml文件

 <?xml version="1.0" encoding="utf-8"?>
 <StudentList>
   <Student Id="1">
     <Name>Tim</Name>
   </Student>
   <Student Id="2">
     <Name>Tom</Name>
   </Student>
   <Student Id="3">
     <Name>Vina</Name>
   </Student>
   <Student Id="4">
     <Name>Emily</Name>
   </Student>
 </StudentList>

xaml:

 <Window x:Class="WpfApplication1.MainWindow"
         xmlns="http://schemas.microsoft.com/winfx/2006/xaml/presentation"
         xmlns:x="http://schemas.microsoft.com/winfx/2006/xaml"
         Title="Simple Binding" Height="206" Width="250">
     <StackPanel x:Name="stackPanel" Background="LightBlue">
         <ListView x:Name="listViewStudents" Height="130" Margin="5">
             <ListView.View>
                 <GridView>
                     <GridViewColumn Header="Name" Width="80" DisplayMemberBinding="{Binding XPath=@Id}" />
                     <GridViewColumn Header="Age" Width="60" DisplayMemberBinding="{Binding XPath=Name}" />
                 </GridView>
             </ListView.View>
         </ListView>
         <Button Content="Load" Height="25" Margin="5" Click="Button_Click" />
     </StackPanel>
 </Window>

这里"{Binding XPath=@Id}"的@表示XML元素的Attribute,而没有@说明是子集元素

xaml.cs:

 using System;
 using System.Collections.Generic;
 using System.Linq;
 using System.Text;
 using System.Threading.Tasks;
 using System.Windows;
 using System.Windows.Controls;
 using System.Windows.Data;
 using System.Windows.Documents;
 using System.Windows.Input;
 using System.Windows.Media;
 using System.Windows.Media.Imaging;
 using System.Windows.Navigation;
 using System.Windows.Shapes;
 using System.Data;
 using MySql.Data;
 using MySql.Data.Entity;
 using MySql.Data.MySqlClient;
 using System.Xml;
 using System.Xml.Linq;

 namespace WpfApplication1
 {
     /// <summary>
     /// Interaction logic for MainWindow.xaml
     /// </summary>
     public partial class MainWindow : Window
     {
         public MainWindow()
         {
             InitializeComponent();
         }

         private void Button_Click(object sender, RoutedEventArgs e)
         {
             XmlDataProvider xdp = new XmlDataProvider();
             xdp.Source = new Uri(@"C:\Users\Administrator\Desktop\Demo\Student.XML");
             xdp.XPath = @"/StudentList/Student";
             this.listViewStudents.DataContext = xdp;
             this.listViewStudents.SetBinding(ListView.ItemsSourceProperty, new Binding());
         }
     }
 }

无论是list,还是DataTable,还是XML文件,我们都可以用LINQ技术来写source,这里就不介绍了,具体可以看书

(七)ObjectDataProvider为source

ObjectDataProvider是为第一种情况的补充,因为不能保证每个类的属性都是暴露在外面可以绑定的,有的时候控件上要绑定的值只是一个类的某个方法里的一些参数,这种情况下第一种方法就没用了,而如果重新设计底层类的风险又太大,这个时候用ObejectDataProvider会比较合适。

现在有一个Calculator的类

 using System;
 using System.Collections.Generic;
 using System.Linq;
 using System.Text;
 using System.Threading.Tasks;

 namespace WpfApplication1
 {
     class Calculator
     {
         public string Add(string arg1, string arg2)
         {
             ;
             ;
             ;
             if (double.TryParse(arg1, out x) && double.TryParse(arg2, out y))
             {
                 z = x + y;
                 return z.ToString();
             }
             return "Input Error!";
         }
     }
 }

在xaml里加个button(这里省略不写了)

在cs里可以先写个简单的objectDataProvider

 using System;
 using System.Collections.Generic;
 using System.Linq;
 using System.Text;
 using System.Threading.Tasks;
 using System.Windows;
 using System.Windows.Controls;
 using System.Windows.Data;
 using System.Windows.Documents;
 using System.Windows.Input;
 using System.Windows.Media;
 using System.Windows.Media.Imaging;
 using System.Windows.Navigation;
 using System.Windows.Shapes;
 using System.Data;
 using MySql.Data;
 using MySql.Data.Entity;
 using MySql.Data.MySqlClient;
 using System.Xml;
 using System.Xml.Linq;

 namespace WpfApplication1
 {
     /// <summary>
     /// Interaction logic for MainWindow.xaml
     /// </summary>
     public partial class MainWindow : Window
     {
         public MainWindow()
         {
             InitializeComponent();
         }

         private void Button_Click(object sender, RoutedEventArgs e)
         {
             ObjectDataProvider odp = new ObjectDataProvider();
             odp.ObjectInstance = new Calculator();
             odp.MethodName = "Add";
             odp.MethodParameters.Add(");
             odp.MethodParameters.Add(");
             MessageBox.Show(odp.Data.ToString());
         }
     }
 }

这里将对象放在ObjectDataProvider.ObjectInstance属性里,将Method放在ObjectDataProvider.MethodName里,而参数得要用ObjectDataProvider.MethodParameters.Add方法来添加,最后它的结果放在ObjectDataProvider.Data里,这里的参数也可以用data binding的方法。

xaml:

 <Window x:Class="WpfApplication1.MainWindow"
         xmlns="http://schemas.microsoft.com/winfx/2006/xaml/presentation"
         xmlns:x="http://schemas.microsoft.com/winfx/2006/xaml"
         Title="Simple Binding" Height="135" Width="300">
     <StackPanel x:Name="stackPanel" Background="LightBlue">
         <TextBox x:Name="textBoxArg1" Margin="5" />
         <TextBox x:Name="textBoxArg2" Margin="5" />
         <TextBox x:Name="textBoxResult" Margin="5" />
     </StackPanel>
 </Window>

cs:

 using System;
 using System.Collections.Generic;
 using System.Linq;
 using System.Text;
 using System.Threading.Tasks;
 using System.Windows;
 using System.Windows.Controls;
 using System.Windows.Data;
 using System.Windows.Documents;
 using System.Windows.Input;
 using System.Windows.Media;
 using System.Windows.Media.Imaging;
 using System.Windows.Navigation;
 using System.Windows.Shapes;
 using System.Data;
 using MySql.Data;
 using MySql.Data.Entity;
 using MySql.Data.MySqlClient;
 using System.Xml;
 using System.Xml.Linq;

 namespace WpfApplication1
 {
     /// <summary>
     /// Interaction logic for MainWindow.xaml
     /// </summary>
     public partial class MainWindow : Window
     {
         public MainWindow()
         {
             InitializeComponent();
             this.SetBinding();
         }

         private void SetBinding()
         {
             ObjectDataProvider odp = new ObjectDataProvider();
             odp.ObjectInstance = new Calculator();
             odp.MethodName = "Add";
             odp.MethodParameters.Add(");
             odp.MethodParameters.Add(");

             Binding bindingToArg1 = new Binding("MethodParameters[0]")
             {
                 Source = odp,
                 BindsDirectlyToSource = true,
                 UpdateSourceTrigger = UpdateSourceTrigger.PropertyChanged
             };

             Binding bindingToArg2 = new Binding("MethodParameters[1]")
             {
                 Source = odp,
                 BindsDirectlyToSource = true,
                 UpdateSourceTrigger = UpdateSourceTrigger.PropertyChanged
             };

             this.textBoxArg1.SetBinding(TextBox.TextProperty, bindingToArg1);
             this.textBoxArg2.SetBinding(TextBox.TextProperty, bindingToArg2);
             this.textBoxResult.SetBinding(TextBox.TextProperty, new Binding(".") { Source = odp });
         }
     }
 }

这里的odp.MethodParameters.Add("0");不一定是0,可以换成其他数字,区别就在于开始的时候是显示其他数字了。

BindsDirectlyToSource = true这句的意思是告诉Binding对象只负责把从UI元素收集到的数据写入其直接Source(即ObjectDataProvider对象)而不是被ObjectDataProvider对象包装着的Calculator对象。

一般情况下,数据从哪里来哪里就是Binding的source,数据到哪里去哪里就应该是binding的target。按这个理论,前两个textbox应该是ObjectDataProvider对象的数据源,而ObjectDataProvider对象又是最后一个TextBox的数据源。但实际上,三个TextBox都以ObjectDataProvider对象为数据源,只是前两个TextBox在Binding的数据流向上做了限制。这样做的原因不外乎有两个:

1. ObjectDataProvider的MethodParameters不是依赖属性,不能作为Binding的目标。

2. 数据驱动UI的理念要求尽可能地使用数据对象作为Binding的Source而把UI元素当做Binding的Target。

(八)RelativeSource为Source

不能确定Source的对象叫什么名字,但知道它与作为Binding目标的对象在UI布局上有相对关系,比如控件自己关联自己的某个数据,关联自己某级容器的数据,这个时候我们就要使用Binding的RelativeSource属性。这里具体的代码就不写了,可以看书

.NET: WPF Data Binding的更多相关文章

  1. WPF QuickStart系列之数据绑定(Data Binding)

    这篇博客将展示WPF DataBinding的内容. 首先看一下WPF Data Binding的概览, Binding Source可以是任意的CLR对象,或者XML文件等,Binding Targ ...

  2. Data Binding in WPF

    http://msdn.microsoft.com/en-us/magazine/cc163299.aspx#S1   Data Binding in WPF John Papa Code downl ...

  3. [WPF]如何调试Data Binding

    前言 在WPF开发中,将ViewModel中对象绑定到UI上时,会出现明明已经将数据对象Binding到UI,但是UI上就是不显示等等的问题.这篇博客将介绍WPF Data Binding相关的内容, ...

  4. Data Binding和INotifyPropertyChanged是如何协调工作的?

    前言 WPF的一大基础就是Data Binding.在基于MVVM架构的基础上,只有通过实现INotifyPropertyChanged接口的ViewModel才能够用于Data Binding. 要 ...

  5. Optimizing Performance: Data Binding(zz)

    Optimizing Performance: Data Binding .NET Framework 4.5 Other Versions   Windows Presentation Founda ...

  6. WPF binding<一> Data Binding在WPF中的地位

    在代码中看到 <Image Source="{Binding ElementName=LBoxImages, Path=SelectedItem.Source}" /> ...

  7. WPF之数据绑定Data Binding

    一般情况下,应用程序会有三层结构:数据存储层,数据处理层(业务逻辑层),数据展示层(UI界面). WPF是“数据驱动UI”. Binding实现(通过纯C#代码) Binding分为source和ta ...

  8. WPF中的Data Binding调试指南

    大家平时做WPF开发,相信用Visual studio的小伙伴比较多.XAML里面曾经在某些特殊版本的Visual Studio中是可以加断点进行调试的,不过目前多数版本都不支持在XAML加断点来调试 ...

  9. WPF教程三:学习Data Binding把思想由事件驱动转变为数据驱动

    之前大家写代码都喜欢用事件驱动,比如说鼠标输入的click事件.初始化的内容全部放在窗体加载完毕的load事件,等等,里面包含了大量的由事件触发后的业务处理代码.导致了UI和业务逻辑高度耦合在一个地方 ...

随机推荐

  1. 弹窗文件js+css

    // 每个弹窗的标识 var x =0; var idzt = new Array(); var Window = function(config){ //ID不重复 idzt[x] = " ...

  2. github中国版本coding.net 的部署和使用

    1.在coding.net注册帐号. 2.安装github,自己百度github软件然后安装. 3.打开coding.net 输入帐号后新建项目 创建项目 创建后,创建ssh公钥,如果不创建的话,在每 ...

  3. zepto源码--整体框架--学习笔记

    为了深入学习javascript,根据别人推荐的方法之一:研究源码. 相对而言,之前的项目中仅仅使用过zepto和jquery,当前阶段,看到好几千行的jquery源码,心生敬畏,望而却步,所以选择相 ...

  4. C/C++链表操作(面试)

    1.为了反转这个单链表,我们先让头结点的next域指向结点2,再让结点1的next域指向结点3,最后将结点2的next域指向结点1,就完成了第一次交换,顺序就变成了Header-结点2-结点1-结点3 ...

  5. HTTP访问的两种方式(HttpClient+HttpURLConnection)整合汇总对比

    HttpClient: HttpClient是Apache Jakarta Common下的子项目,用来提供高效的.最新的.功能丰富的支持HTTP协议的客户端编程工具包,并且它支持HTTP协议最新的版 ...

  6. 转:[ASP.NET]重構之路系列v4 – 簡單使用interface之『你也會IoC』

    前言 上次v3版本,我們將Entity, Service, Dao, Utility都放到了類別庫裡面,讓我們可以輕鬆的在不同專案中用同一份組件.雖然文章沒有獲得太多的讚賞,不過相信那一定是太多人會這 ...

  7. 关于前后台交互生成json区别

    如何返回[object{xx:{}}]这种数组型json在服务器端return $arr[]=m;像这种都可以产生[Object { 0="9", 1="8", ...

  8. Selenium2学习-015-WebUI自动化实战实例-013-通过 URL 关闭多余的已开浏览器窗口

    在日常的 WebUI 自动化测试脚本执行的过程中,经常会打开不同的网页,进行相应的操作,此时可能会打开很多的网页,当打开的网页过多时,无效的网页资源对运行脚本的机器造成了过多无效的资源浪费,因而在日常 ...

  9. each的用法

    $(selector).each(function(index,element)) function(index,element) 必需.为每个匹配元素规定运行的函数. index - 选择器的 in ...

  10. Linux版OpenVPN安装、配置教程(转)

    本文将以目前最新的openvpn-2.3.4.tar.gz(更新于2014-5-2,下载地址)为例来介绍如何在Linux系统中安装.配置及使用OpenVPN. 在这里,我们选用了一台预装CentOS ...