WPF實現(xiàn)數(shù)據(jù)綁定
簡單而言, 數(shù)據(jù)綁定是一種關(guān)系, 這種關(guān)系告訴WPF 從一個源目標(biāo)對象中提取一些信息, 并且使用該信息設(shè)置為目標(biāo)對象的屬性。目標(biāo)屬性總是依賴項屬性, 并且通常位于WPF元素中。
然而, 源對象可以是任何內(nèi)容, 可是是隨機(jī)生成的一個對象、也可以是數(shù)據(jù)庫的數(shù)據(jù)對象,或者手動創(chuàng)建的對象。
簡單綁定
為了能夠簡單理解這種綁定關(guān)系, 接下來簡單示例, 用一個數(shù)值滾動條, 動態(tài)修改一個文字的字體大小, 通過綁定的方式。
<StackPanel> <Slider Name="s1" Value="10" Maximum="100"></Slider> <TextBlock FontSize="{Binding ElementName=s1,Path=Value}" Text="看著我" ></TextBlock> </StackPanel>
上圖中, TextBlock的字體大小FontSize通過綁定的形式與名稱為s1的元素的Value屬性綁定。
注:數(shù)據(jù)綁定表達(dá)式使用了XAML標(biāo)記擴(kuò)展, 所以綁定表達(dá)式以單詞Binding 開頭。該示例中設(shè)置的兩個屬性: ElementName(源對象元素) 和Path(源對象元素中的屬性)。
運(yùn)行測試效果:
綁定錯誤
WPF不會引發(fā)異常來通知與數(shù)據(jù)綁定相關(guān)的問題, 如果指定的元素不存在或錯誤, 不會收到任何提示, 最簡單的只是目標(biāo)的屬性無法發(fā)生改變, 不過, WPF仍然會在輸出窗口顯示其綁定失敗的具體細(xì)節(jié)。
綁定模式
在上面的示例中, 只是一個最簡單的綁定, 我們可以假設(shè)一個環(huán)境, 在上面的例子基礎(chǔ)上另外加一個按鈕, 去單獨(dú)實現(xiàn)字體的大小設(shè)置, 然而拖動條并不會隨著按鈕設(shè)置的字體大小而改動。
但是在實際應(yīng)用中, 我們甚至?xí)玫降亩喾N綁定模式, 所以在WPF中, 含有5種綁定的模式, 包含在System.Windows.Data.BindingMode枚舉中。
OneWay | 當(dāng)源屬性變化時更新目標(biāo)屬性 |
TwoWay | 當(dāng)源數(shù)據(jù)變化時更新目標(biāo)屬性, 并且當(dāng)目標(biāo)屬性變化時更新源屬性 |
OneTime | 最初根據(jù)源屬性值設(shè)置目標(biāo)屬性。然而, 在此之后的所有改變都會被忽略(除非綁定被設(shè)定為一個完全不同的對象或者調(diào)用BindingExpression.UpdateTarget()方法, 如稍后所介紹的那樣)。通常, 如果知道源屬性不會變化, 可以使用這種模式降低開銷。 |
OneWayToSource | 和OneWay 類似, 但是方向相反。當(dāng)目標(biāo)屬性變化時更新源屬性(這看起來有點像向后傳遞),但目標(biāo)屬性永遠(yuǎn)不會更新。 |
Default | 此類綁定依賴于目標(biāo)屬性。它既可以是雙向的(對于用戶設(shè)置的屬性, 如Textbox.Text屬性),也可以是單向的(對于所有其他屬性)。除非明確指定了另外一種模式, 否則所有綁定都使用該方法。 |
在此, 對于OneTime設(shè)置過一次就無效,和默認(rèn)default模式就不過多介紹, 下面對其他集中進(jìn)行簡單分析:
利用代碼進(jìn)行數(shù)據(jù)綁定
下面代碼演示如何將上面的示例進(jìn)行綁定:
private void Window_Loaded(object sender, RoutedEventArgs e) { Binding bind = new Binding(); bind.Source = s1; //指定源對象 bind.Mode = BindingMode.Default; //設(shè)置綁定模式 bind.Path = new PropertyPath("Value"); //關(guān)聯(lián)綁定的屬性 txtfont.SetBinding(TextBlock.FontSizeProperty, bind); //綁定對象 }
下面代碼演示如何解除數(shù)據(jù)綁定:
private void Window_Loaded(object sender, RoutedEventArgs e) { //刪除指定目標(biāo)對象的 指定屬性數(shù)據(jù)綁定 BindingOperations.ClearBinding(txtfont, TextBlock.FontSizeProperty); //刪除指定目標(biāo)對象的所有數(shù)據(jù)綁定 BindingOperations.ClearAllBindings(txtfont); }
注意:以上的兩個移除數(shù)據(jù)綁定的方法, 可以刪除任何形式的綁定, 無論是代碼創(chuàng)建的綁定還是通過XAML標(biāo)記的綁定。
綁定更新:
在上面的示例中, 從 源對象 -> 目標(biāo)對象, 源對象的值發(fā)生改變, 目標(biāo)會立刻響應(yīng)。但是從目標(biāo) > 源 , 未必會立即發(fā)生。
在WPF中,他們的行為右 binding中的 UpdateSourceTrigger屬性控制, 關(guān)于UpdateSourceTrigger 下面列出了對應(yīng)的枚舉值
propertyChanged | 當(dāng)目標(biāo)屬性發(fā)生變化時立即更新源目標(biāo) |
LostFocus | 當(dāng)目標(biāo)屬性發(fā)生變化時并且目標(biāo)丟失焦點是更新源目標(biāo) |
Explicit | 除非調(diào)用BindingExpression.UpdateSource()方法,否則無法更新資源 |
Default | 根據(jù)目標(biāo)屬性的元數(shù)據(jù)確定更新行為, 大多數(shù)屬性的默認(rèn)行為是PropertyChanged, 但是TextBox.Text的屬性默認(rèn)行為是LoastFocus |
示例, 對于TextBox控件, 添加 UpdateSourceTrigger=PropertyChanged 以達(dá)到更新源目標(biāo), 但是往往在實際應(yīng)用中, TextBox會頻繁的變化, 從而導(dǎo)致多次更新源目標(biāo),
PropertyChenged更新反而會是應(yīng)用程序運(yùn)行更加緩慢, 要完全控制源對象的更新時機(jī), 可以選擇 Explicit模式, 例如, 添加一個按鈕, 在按鈕中調(diào)用UpdateSource方法已達(dá)到
更新的目的。
<TextBlock Grid.Row="1" Text="看著我" Name="txtfont"></TextBlock> <TextBox Grid.Row="2" Margin="10" Text="{Binding ElementName=txtfont,Path=FontSize,UpdateSourceTrigger=PropertyChanged}"></TextBox>
注:在調(diào)用UpdateSource之前, 需要通過GetBindingExpression()方法獲取到BindingExpression對象, 從而調(diào)用UpdateSource()方法。
private void Button_Click(object sender, RoutedEventArgs e) { BindingExpression exp = txtfont.GetBindingExpression(TextBlock.FontSizeProperty); exp.UpdateSource(); }
非元素綁定:
在上面的例子中, 一直圍繞著元素之間的綁定, 以及一些綁定的模式的介紹, 但是在我們的實際應(yīng)用當(dāng)中, 更多的是于數(shù)據(jù)打交道, 就像常見的Grid表格, 需要綁定對應(yīng)數(shù)據(jù)庫表的每個字段。
當(dāng)綁定一個非元素對象時候, 這個時候就不能使用 ElementName屬性, 在WPF中, 提供了多種屬性以在實際應(yīng)用中選擇:
Source | 提供數(shù)據(jù)的對象本身 |
RelativeSource | 使用RelativeSource對象指向源目標(biāo), |
DataContext | 從當(dāng)前元素向下, 找到第一個非空的DataContext屬性。 |
Source屬性:
如下, 我們定義了一個Custom的資源對象, 同時創(chuàng)建了一個TextBlock對象用Source進(jìn)行了對象綁定, 此時, TextBlock顯示的Text則為Tom
<Window.Resources> <FontFamily x:Key="Custom">Tom</FontFamily> </Window.Resources> <StackPanel> <TextBlock Text="{Binding Source={StaticResource Custom},Path=Source}" Height="30"/> </StackPanel>
RelativeSource屬性:
如下, 定義了一個RelativeSource對象, Mdoe選擇了FindAncestor(該模式通知元素直到發(fā)現(xiàn)AncestorType定義的元素類型,如下類型為 window),選擇的Paht為title屬性
<StackPanel> <TextBlock Text="{Binding RelativeSource={RelativeSource Mode=FindAncestor,AncestorType={x:Type Window}},Path=Title}" Height="30"/> </StackPanel>
DataContext屬性:
TextBlock綁定了DataContext屬性, WPF會在元素樹種,找到第一個不為null的數(shù)據(jù)上下文, 此時后臺需要對DataContext進(jìn)行綁定指定的上下文對象。
<TextBlock Text="{Binding Width}"/>
該示例種, 綁定了自身為上下文對象, 所有TextBlock找到的第一個Width屬性為windows窗體本身的Width屬性
private void Window_Loaded(object sender, RoutedEventArgs e) { this.DataContext = this; }
到此這篇關(guān)于WPF實現(xiàn)數(shù)據(jù)綁定的文章就介紹到這了。希望對大家的學(xué)習(xí)有所幫助,也希望大家多多支持腳本之家。
相關(guān)文章
MVC+EasyUI+三層新聞網(wǎng)站建立 主頁布局的方法(五)
這篇文章主要為大家詳細(xì)介紹了MVC+EasyUI+三層新聞網(wǎng)站建立的第五篇,教大家如何進(jìn)行主頁布局,具有一定的參考價值,感興趣的小伙伴們可以參考一下2017-07-07ASP.NET筆記之 ListView 與 DropDownList的使用
本篇文章小編為大家介紹,ASP.NET筆記之 ListView 與 DropDownList的使用。需要的朋友參考下2013-04-04ASP.NET網(wǎng)站導(dǎo)航及導(dǎo)航控件如何使用
這篇文章主要介紹了ASP.NET網(wǎng)站導(dǎo)航及導(dǎo)航控件如何使用,需要的朋友可以參考下2015-09-09