一步一步学Silverlight 2系列(11):数据绑定

概念

Silverlight 2 Beta 1版本发布了,无论从Runtime还是Tools都给我们带来了很多的惊喜,如支持框架语言Visual Basic, Visual C#, IronRuby, Ironpython,对JSON、Web Service、WCF以及Sockets的支持等一系列新的特性。《一步一步学Silverlight 2系列》文章带您快速进入Silverlight 2开发。

本文为系列文章第十一篇,主要介绍Silverlight 2中的数据绑定。

数据绑定模式

在Silverlight 2中,支持三种模式的数据绑定。

1.OneTime:一次绑定,在绑定创建时使用源数据更新目标,适用于只显示数据而不进行数据的更新。

2.OneWay:单向绑定,在绑定创建时或者源数据发生变化时更新到目标,适用于显示变化的数据。

3.TwoWay:双向绑定,在任何时候都可以同时更新源数据和目标。

Jesse Liberty举的例子非常的形象,使用Silverlight开发一个在线书店,显示书籍的书名、作者等信息,使用OneTime模式,这些数据一般不会发生变化的;显示价格信息时使用OneWay模式,因为管理员可能会在一天内调整价格;显示书籍的剩余数量时用TwoWay模式,数量随着用户的订购会随时发生变化,即目标和源数据都要进行更新。

简单数据绑定

在本示例中我们将做一个简单的数据绑定,用来显示用户信息,XAML如下:

<Grid x:Name="LayoutRoot" Background="#46461F">
    <Grid.RowDefinitions>
        <RowDefinition Height="160"></RowDefinition>
        <RowDefinition Height="40"></RowDefinition>
        <RowDefinition Height="40"></RowDefinition>
    </Grid.RowDefinitions>
    <Grid.ColumnDefinitions>
        <ColumnDefinition Width="150"></ColumnDefinition>
        <ColumnDefinition Width="*"></ColumnDefinition>
    </Grid.ColumnDefinitions>
    <Image Source="terrylee.jpg" Width="78" Height="100"
           HorizontalAlignment="Left" Grid.Row="0" Grid.Column="1"/>
    <TextBlock Foreground="White" FontSize="18" Text="姓名:"
               Grid.Row="1" Grid.Column="0" HorizontalAlignment="Right"/>
    <TextBlock x:Name="lblName" Foreground="White" FontSize="18"
               Grid.Row="1" Grid.Column="1" HorizontalAlignment="Left"/>
    <TextBlock Foreground="White" FontSize="18" Text="位置:"
               Grid.Row="2" Grid.Column="0" HorizontalAlignment="Right"/>
    <TextBlock x:Name="lblAddress" Foreground="White" FontSize="18"
               Grid.Row="2" Grid.Column="1" HorizontalAlignment="Left"/>
</Grid>

添加一个简单User类,它具有Name和Address两个属性:

public class User
{
    public string Name { get; set; }

    public string Address { get; set; }
}

使用绑定句法{Binding Property}进行数据绑定,注意下面的两个TextBlock控件Text属性:

<Grid x:Name="LayoutRoot" Background="#46461F">
    <Grid.RowDefinitions>
        <RowDefinition Height="160"></RowDefinition>
        <RowDefinition Height="40"></RowDefinition>
        <RowDefinition Height="40"></RowDefinition>
    </Grid.RowDefinitions>
    <Grid.ColumnDefinitions>
        <ColumnDefinition Width="150"></ColumnDefinition>
        <ColumnDefinition Width="*"></ColumnDefinition>
    </Grid.ColumnDefinitions>
    <Image Source="terrylee.jpg" Width="78" Height="100"
       HorizontalAlignment="Left" Grid.Row="0" Grid.Column="1"/>
    <TextBlock Foreground="White" FontSize="18" Text="姓名:"
           Grid.Row="1" Grid.Column="0" HorizontalAlignment="Right"/>
    <TextBlock x:Name="lblName" Foreground="White" FontSize="18"
               Grid.Row="1" Grid.Column="1" HorizontalAlignment="Left"
               Text="{Binding Name}"/>
    <TextBlock Foreground="White" FontSize="18" Text="位置:"
               Grid.Row="2" Grid.Column="0" HorizontalAlignment="Right"/>
    <TextBlock x:Name="lblAddress" Foreground="White" FontSize="18"
               Grid.Row="2" Grid.Column="1" HorizontalAlignment="Left"
               Text="{Binding Address}"/>
</Grid>

指定数据源,注意这里是创建一个User的实例并赋值后,把user实例绑定到了TextBlock的DataContext上,而不是向之前我们所做的示例中那样,直接指定Text属性:

private void UserControl_Loaded(object sender, RoutedEventArgs e)
{
    User user = new User();
    user.Name = "TerryLee";
    user.Address = "中国 天津";

    lblName.DataContext = user;
    lblAddress.DataContext = user;
}

运行示例后,可以看到:

TerryLee_Silverlight2_0055

上面这种数据绑定模式,只是显示数据而不对数据做任何修改,默认的绑定模式是一次绑定OneTime。

单向绑定示例

如果需要在数据源发生变化时能够通知UI进行相应的更新,即使用单向绑定OneWay或者双向绑定TwoWay,则业务实体需要实现接口INotifyPropertyChanged。在本示例中,我们加上一个更新按钮,当单击按钮时更新user实例的属性值,会看到界面上的数据也会发生变化。

修改一下User类,使其实现INotifyPropertyChanged接口。

public class User : INotifyPropertyChanged
{
    public event PropertyChangedEventHandler PropertyChanged;

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

    private string _address;
    public string Address
    {
        get { return _address; }
        set 
        {
            _address = value;
            if (PropertyChanged != null)
            {
                PropertyChanged(this, new PropertyChangedEventArgs("Address"));
            }
        }
    }
}

修改数据绑定模式,使用单向绑定OneWay模式,如{Binding Address, Mode=OneWay}

<Grid x:Name="LayoutRoot" Background="#46461F">
    <Grid.RowDefinitions>
        <RowDefinition Height="160"></RowDefinition>
        <RowDefinition Height="40"></RowDefinition>
        <RowDefinition Height="40"></RowDefinition>
    </Grid.RowDefinitions>
    <Grid.ColumnDefinitions>
        <ColumnDefinition Width="150"></ColumnDefinition>
        <ColumnDefinition Width="*"></ColumnDefinition>
    </Grid.ColumnDefinitions>
    <Image Source="terrylee.jpg" Width="78" Height="100"
       HorizontalAlignment="Left" Grid.Row="0" Grid.Column="1"/>
    <Button x:Name="btnUpdate" Width="100" Height="40"
            Content="Update" Click="btnUpdate_Click"/>
    <TextBlock Foreground="White" FontSize="18" Text="姓名:"
           Grid.Row="1" Grid.Column="0" HorizontalAlignment="Right"/>
    <TextBlock x:Name="lblName" Foreground="White" FontSize="18"
               Grid.Row="1" Grid.Column="1" HorizontalAlignment="Left"
               Text="{Binding Name, Mode=OneWay}"/>
    <TextBlock Foreground="White" FontSize="18" Text="位置:"
               Grid.Row="2" Grid.Column="0" HorizontalAlignment="Right"/>
    <TextBlock x:Name="lblAddress" Foreground="White" FontSize="18"
               Grid.Row="2" Grid.Column="1" HorizontalAlignment="Left"
               Text="{Binding Address, Mode=OneWay}"/>
</Grid>

编写事件处理程序,为了演示把user声明为一个全局的,并在按钮的单击事件中修改其属性值:

public partial class Page : UserControl
{
    public Page()
    {
        InitializeComponent();
    }
    User user;
    private void UserControl_Loaded(object sender, RoutedEventArgs e)
    {
        user = new User();
        user.Name = "TerryLee";
        user.Address = "中国 天津";

        lblName.DataContext = user;
        lblAddress.DataContext = user;
    }

    private void btnUpdate_Click(object sender, RoutedEventArgs e)
    {
        user.Name = "李会军";
        user.Address = "China Tianjin";
    }
}

运行后如下所示:

TerryLee_Silverlight2_0056

单击Update按钮后:

TerryLee_Silverlight2_0057

绑定到列表

下面再看一个绑定到列表的简单例子,一般都会使用DataGrid或者ListBox来进行列表数据的显示。下面的示例我们显示一个文章列表:

<Grid Background="#46461F">
    <Grid.RowDefinitions>
        <RowDefinition Height="40"></RowDefinition>
        <RowDefinition Height="*"></RowDefinition>
    </Grid.RowDefinitions>
    <Grid.ColumnDefinitions>
        <ColumnDefinition></ColumnDefinition>
    </Grid.ColumnDefinitions>
    <Border Grid.Row="0" Grid.Column="0" CornerRadius="15"
            Width="240" Height="36" Background="Orange"
            Margin="20 0 0 0" HorizontalAlignment="Left">
        <TextBlock Text="文章列表" Foreground="White"
                   HorizontalAlignment="Left" VerticalAlignment="Center"
                   Margin="20 0 0 0"></TextBlock>
    </Border>
    <ListBox x:Name="PostList" Grid.Column="0" Grid.Row="1"
             Margin="40 10 10 10"
             HorizontalContentAlignment="Left" VerticalContentAlignment="Bottom"
             ItemsSource="{Binding Posts}">
    </ListBox>
</Grid>

编写一个简单的业务类:

public class Blog
{
    public List<String> Posts { get; set; }
}

初始化集合数据并进行绑定

private void UserControl_Loaded(object sender, RoutedEventArgs e)
{
    Blog blog = new Blog();
    blog.Posts = new List<String>
    {
        "一步一步学Silverlight 2系列(10):使用用户控件",
        "一步一步学Silverlight 2系列(9):使用控件模板",
        "一步一步学Silverlight 2系列(8):使用样式封装控件观感",
        "一步一步学Silverlight 2系列(7):全屏模式支持"
    };

    PostList.DataContext = blog;
}

最终运行的结果如下所示:

TerryLee_Silverlight2_0058

当然我们也可以使用ListBox的ItemsSource属性进行绑定,

结束语

本文简单介绍了Silverlight 2中的数据绑定,你可以从这里下载文章示例代码。

下一篇:一步一步学Silverlight 2系列(12):数据与通信之WebClient

作者:TerryLee
出处:http://terrylee.cnblogs.com
本文版权归作者和博客园共有,欢迎转载,但未经作者同意必须保留此段声明,且在文章页面明显位置给出原文连接,否则保留追究法律责任的权利。
Tag标签: Silverlight
posted @ 2008-03-08 23:48 TerryLee 阅读(13825) 评论(83)  编辑 收藏 所属分类: [03]  银光点亮世界

  回复  引用  查看    
#1楼2008-03-09 01:12 | Jacky_Xu      
辛苦了。已经写两天了!都有点跟不上了! :)
  回复  引用  查看    
#2楼2008-03-09 01:22 | hxm      
  
#3楼 |

  
#4楼 |

  回复  引用  查看    
#5楼[楼主]2008-03-09 12:24 | TerryLee      
@Jacky_Xu
呵呵,不辛苦:)

  回复  引用  查看    
#6楼[楼主]2008-03-09 12:24 | TerryLee      
@hxm
:)

  回复  引用  查看    
#7楼[楼主]2008-03-09 12:28 | TerryLee      
@过客007
Silverlight不可以直接访问数据,你可以通过WCF、JSON、WebClient、WebRequest等方式取数据,并进行绑定,但是支持LINQ和LINQ to XML,后面会有相关的例子

  回复  引用  查看    
#8楼[楼主]2008-03-09 12:29 | TerryLee      
@过客007
现在看来只能这样去做,一个项目生成一个.xap文件

  
#9楼 |

  回复  引用  查看    
#10楼[楼主]2008-03-09 15:18 | TerryLee      
@过客007
不用客气^_^

  回复  引用  查看    
#11楼2008-03-10 22:09 | 守護~︶箬      
李老师,我调试的时候经常出现错误:文件“D:\Silverlight\ControlCustom\ControlCustom\ClientBin\ControlCustom.xap”正由另一进程使用,因此该进程无法访问该文件。

是哪个进程呢?

  回复  引用  查看    
#12楼[楼主]2008-03-10 22:13 | TerryLee      
@守護~︶箬
你把Web Development Server停止了,然后再调试,试试看

  回复  引用  查看    
#13楼2008-03-10 23:09 | ArthurHend      
请问private void UserControl_Loaded()这个是写在哪个文件中的?
  回复  引用  查看    
#14楼[楼主]2008-03-10 23:19 | TerryLee      
@ArthurHend
Page.xaml.cs文件中的,可以下载示例代码看一下:)

  回复  引用  查看    
#15楼2008-03-10 23:27 | ArthurHend      
@TerryLee
刚下来看过了!呵呵……我想问这个UserControl_Loaded是哪里自动生成的还是直接写的啊?

  回复  引用  查看    
#16楼[楼主]2008-03-10 23:41 | TerryLee      
在可以在XAML中编辑器中,未UserControl添加Loaded事件,按Tab键自动生成,也可以自己手动去写。可以看一下鼠标事件处理一文:
http://www.cnblogs.com/Terrylee/archive/2008/03/07/Silverlight2-step-by-step-part4-mouse-events-handle.html" target="_new">http://www.cnblogs.com/Terrylee/archive/2008/03/07/Silverlight2-step-by-step-part4-mouse-events-handle.html

  
#17楼 |

  回复  引用  查看    
#18楼[楼主]2008-03-13 12:58 | TerryLee      
@高
可能是你的开发环境哪儿安装的有问题吧

  回复  引用  查看    
#19楼2008-03-13 20:17 | Zhuang miao      
你写的比我看的还快!是人是神??
  回复  引用  查看    
#20楼[楼主]2008-03-13 20:20 | TerryLee      
@Zhuang miao
是人,呵呵:)

  回复  引用  查看    
#21楼2008-03-13 20:27 | Zhuang miao      
@TerryLee
这两天一直在看你写的这个系列,真不错,就是看的有点慢~那个WCF是你翻译的吗?准备买本

  回复  引用  查看    
#22楼[楼主]2008-03-13 20:35 | TerryLee      
@Zhuang miao
不是,是张逸和徐宁翻译的:)

  回复  引用  查看    
#23楼2008-03-20 12:23 | 张荣华      
@TerryLee
UserControl_Loaded()方法是不是需要在Page()中调用一下啊?

  回复  引用  查看    
#24楼[楼主]2008-03-20 12:54 | TerryLee      
@张荣华
在XAML中声明就可以了...

  
#25楼 |

  回复  引用  查看    
#26楼2008-04-19 11:34 | yww      
Silverlight不可以直接访问数据,你可以通过WCF、JSON、WebClient、WebRequest等方式取数据,并进行绑定,但是支持LINQ和LINQ to XML,后面会有相关的例子
--这个后面的例子在哪里?

另外,习惯了DataGrid和GridView的asp.net developer, 似乎在这个grid和ListBox 的例子中中找不到Header,TemplateColumn等概念。如何绑定一个常见的数据表格呢?

  
#27楼 |

  
#28楼 |

  回复  引用  查看    
#29楼2008-06-02 21:01 | tthxnz      
请问绑定的时候如何写格式化字符串呢
  
#30楼 |

  
#31楼 |

  
#32楼 |

  
#33楼 |

  回复  引用  查看    
#34楼2008-07-01 14:44 | 游民一族      
呵呵,遇到两个问题:
1. 把“public class User {...”前面的public去了绑定就报异常,很是奇怪,在同一个程序里public修饰符应该不是必需的吧!
2. 不能绑定到Page上,写了个LayoutRoot.DataContext = this,一运行就报错,郁闷

  回复  引用  查看    
#35楼[楼主]2008-07-01 22:15 | TerryLee      
@游民一族
1.必须为public

2.LayoutRoot.DataContext=this,搞明白这里的this是什么,呵呵

  
#36楼 |

  回复  引用  查看    
#37楼[楼主]2008-08-25 11:27 | TerryLee      
@TM123456
能不能多提供点信息啊,有什么错误信息没?涉及到跨域调用问题没有?

  
#38楼 |

  回复  引用  查看    
#39楼[楼主]2008-10-08 11:19 | TerryLee      
@tienhy
报什么错误?

  
#40楼 |

  回复  引用  查看    
#41楼[楼主]2008-10-24 10:06 | TerryLee      
@傻苗苗
这个有些难度,我不确定SQL Server 2005中有没有主动通知机制。

  
#42楼 |

  
#43楼 |

  回复  引用  查看    
#44楼[楼主]2008-10-25 10:13 | TerryLee      
@傻苗苗
这个是没有问题的,你可以用一个类似于Timer的东西来实现,隔段时间去主动调用Web Service来更新数据。

  
#45楼 |

  
#46楼 |

  回复  引用  查看    
#47楼[楼主]2008-10-28 10:07 | TerryLee      
@John.liu
WCF的调用我也写过一篇文章的。

  回复  引用  查看    
#48楼[楼主]2008-10-28 10:08 | TerryLee      
@傻苗苗
随时欢迎:)

  
#49楼 |

  回复  引用  查看    
#50楼[楼主]2008-10-29 13:24 | TerryLee      
@傻苗苗
现在Silverlight是支持WCF的双工通信的,在开发文档中有相关的示例。

  回复  引用  查看    
#51楼2008-11-06 11:41 | zhaoliang      
你好,求助
我在安装了blend2.5 之后再打开之前VS2008里面的项目,出现如下错误,不知道怎么解决

Error HRESULT E_FAIL has been returned from a call to a COM component.

多谢了!

  回复  引用  查看    
#52楼[楼主]2008-11-06 20:27 | TerryLee      
@zhaoliang
1. 你所说的之前VS2008的项目,是Silverlight哪个版本?

2. 微软好像发布了一个Blend的SP1包,好像是需要安装这个。

  
#53楼 |

  回复  引用  查看    
#54楼[楼主]2008-11-20 23:42 | TerryLee      
@卅
呵呵,太客气了:)

  
#55楼 |

  回复  引用  查看    
#56楼[楼主]2008-12-01 00:05 | TerryLee      
@luolicun
PropertyChanged是我自定义的一个方法而已:)

  
#57楼 |

  
#58楼 |

  
#59楼 |

  回复  引用  查看    
#60楼[楼主]2008-12-08 10:51 | TerryLee      
@garyzhang2681
列表更新重新设置ItemsSource啊

  
#61楼 |

  回复  引用  查看    
#62楼[楼主]2009-01-04 11:27 | TerryLee      
@zhlan
不是的,我有一篇文章《在Silverlight 2中切换用户控件》专门写这个问题,你可以参考一下。

  
#63楼 |

  回复  引用  查看    
#64楼2009-04-28 15:48 | 睡の虫      
不能直接绑XML啊??
  
#65楼 |

  回复  引用  查看    
#66楼2009-05-03 20:12 | 灵魂架构师      
迟到了
不过真的很好!

  回复  引用    
#67楼2009-05-05 15:41 | afaand20      
Error 1 The type or namespace name 'List' could not be found (are you missing a using directive or an assembly reference?) 。我在做绑定到列表出现了这个错误,是不是没有引用什么呀?
  
#68楼 |

  
#69楼 |

  回复  引用  查看    
#70楼2009-06-19 11:36 | 笨笨猪      
会军,关于图片只能放在clientBin下吗?能否自定义配置?还有就是要把sliverlight挂到一个网站,必须建立一个叫clientBin的文件夹吗?能否自定义名字?谢谢
  回复  引用  查看    
#71楼2009-06-19 13:39 | 笨笨猪      
基础终于看完了,谢谢会军:)



发表评论

昵称: [登录] [注册]

主页:

邮箱:(仅博主可见)

评论内容:

  登录  注册

[使用Ctrl+Enter键快速提交评论]

0 1096944


相关文章:

相关链接: