數據banding的最簡單情形是,源對象時WPF元素而且源屬性是依賴性屬性。前面章節解釋過,依賴項屬性具有內置的更改通知支持。因此,當在源對象中改變依賴項屬性的值時,會立即更新目標對象中的綁定屬性。這正是我們所需要的行為——而且不必為此構建任何額外的基礎結構。
為理解如何將一個元素綁定到另一個元素,下面創建一個簡單的示例。該示例窗口包含了兩個控件:一個Slider控件和一個具有單行文本的TextBlock控件。如果向右拖動滑動條上的滑塊,文本字體的尺寸會立即隨之增加。如果向左拖動滑塊,字體尺寸會縮小。

顯然,使用代碼創建這種哦弄個行為不是很難。可簡單地響應Slider.ValueChanged事件,並將滑動條控件的當前值復制到TextBlock控件來實現這種行為。不過,通過數據綁定實現這種行為更簡單。
一、綁定表達式
當使用數據綁定時,不必對源對象(在本例中是Slider控件)做任何改動。只需要配置源對象使其屬性具有正確的值范圍,通常進行如下配置:
<Slider Name="sliderFontSize" Margin="3" Minimum="1" Maximum="40" Value="10" TickFrequency="1" IsSnapToTickEnabled="True" TickPlacement="TopLeft"></Slider>
綁定時在TextBlock元素中定義的。在此沒有使用字面值設置FontSize屬性,而是使用了一個綁定表達式,如下所示:
<TextBlock Margin="10" Name="lblSampleText" FontSize="{Binding ElementName=sliderFontSize, Path=Value}" Text="Simple Text"></TextBlock>
數據綁定表達式使用XAML標記擴展(因此具有花括號)。因為正在創建System.Windows.Data.Binding類的一個實例,所以綁定表達式以單詞Binding開頭。盡管可采用多種方式配置Binding對象,但本例中只需要設置兩個屬性:ElementName屬性(指示源元素)和Path屬性(指示源元素中的屬性)。
之所以使用名稱Path而不是Property,是因為Path可能指向屬性的屬性(如FontFamily.Source),也可能指向屬性使用的索引器(如Content.Children[0])。可構建具有多級層次的路徑,使其指向水屬性的屬性的屬性,一次類推。
如果希望引用附加屬性(在另一個類中定義但應用於綁定元素的屬性),那么需要在圓括號中封裝屬性名稱。例如,如果綁定到Grid控件中的某個元素,路徑(Grid.Row)將檢索放置元素的行的行號。
二、綁定錯誤
WPF不會引發因此來通知與數據綁定相關的問題。如果指定的元素或屬性不存在,那么不會受到任何指示;相反,指示不能在目標屬性中顯示數據。
咋一看,對調試而言這像是可怕的夢魘。幸運的是,WPF輸出了綁定失敗細節的跟蹤信息。當調試應用程序時,該信息顯示在Visual Studio的Output窗口中。
當試圖讀取源屬性時,WPF會忽略拋出的任何異常,並不加提示地丟棄因數據無法轉換為目標屬性的數據類型而引發的異常。然而,當處理這些問題時還有一種選擇——可通知WPF改變源元素的外觀以指示發生了錯誤。例如,當使用哦感嘆號圖標或紅色輪廓標識非法輸入。
三、綁定模式
數據綁定的一個特性是目標會被自動更新,而不考慮源的修改方式。在這個示例中,源只能通過一種方式進行修改——通過用戶與滑動條上滑動進行的交互。下面分析該例的一個稍經修改的版本:添加一個按鈕,每個按鈕為滑動條應用一個預先設置的值:

當單擊Set to Large按鈕時,會運行下面的代碼:
private void cmd_SetLarge(object sender, RoutedEventArgs e) { sliderFontSize.Value = 30; }
上面的代碼設置滑動條的值,這會通過數據綁定強制改變字體大小。效果與移動滑動條上的滑塊一樣。
然而,下面的代碼不能正常工作:
private void cmd_SetLarge(object sender, RoutedEventArgs e) { lblSampleText.FontSize = 30; }
上面的代碼直接設置文本框的字體尺寸。因此,滑動條的位置未響應地更新。更糟的是,上面的代碼破壞了字體尺寸的綁定,並用字面值代替了綁定。如果現在移動滑動條上的滑塊,文本框根本不會響應地進行改變。
有趣的是,可采用一種方式強制在兩個方向傳遞數據:從源到目標以及從目標到源。技巧是設置Binding對象的Mode屬性。下面的是修訂后過的雙向綁定,該綁定允許為源或目標應用變化,並使整體的其他部分自動更新自身:
<TextBlock Margin="10" Name="lblSampleText" FontSize="{Binding ElementName=sliderFontSize, Path=Value,Mode=TwoWay}" Text="Simple Text"></TextBlock>
在這個示例中,沒有理由使用雙向綁定(這需要更大的開銷),因為可通過使用正確的編碼來解決問題。然而,考慮該例的一個變體,該變體包含一個可在其中精確設置字體尺寸的文本框。這個文本框需要使用雙向綁定,從而當通過另一個方法改變字體尺寸時,該文本框可以應用用戶的改變,並顯示最新的尺寸值。
當設置Binding.Mode屬性時,WPF允許使用5個System.Windows.Data.BindingMode枚舉值中的任何一個。下表列出了全部枚舉值:
表 BindingMode枚舉值

下圖顯示了他們之間的區別。前面已經介紹了OneWay和TwoWay模式。OneTime模式非常簡單。下面對其他兩種模式再進行一些分析。

圖 綁定兩個屬性的不同方式
1、OneWayToSource模式
你可能會好奇,既然有了OneWay模式,為什么還有OneWayToSource模式選項——畢竟這兩個值都以相同方式創建單向綁定。唯一區別是綁定表達式的放置位置。本質行,OneWayToSource模式允許通過在通常被視為綁定源的對象中放置綁定表達式,從而翻轉源和目標。
使用這一技巧最常見的原因是要設置非依賴項屬性的屬性。前面開始介紹過,綁定表達式只能用於設置依賴項屬性。但通過使用OneWayToSource模式,可克服這一限制,但前提是提供數據的屬性本身是依賴項屬性。
2、Default模式
最后,除非明確指定其他模式,否則可能認為所有綁定都是單向的,這看起來像是符合邏輯的(畢竟,簡單的滑動條示例使用的就是這種方式)。然而,情況並非如此。為了自我驗證這一事實,在此考慮具有能夠改變字體尺寸的綁定文本框的示例。即使刪除了Mode=TwoWay設置,這個示例也仍工作的很好。這是因為WPF使用了一種不同的、默認情況下依賴於所綁定屬性的模式(從技術角度看,在每個依賴項屬性中都有一個元數據——FrameworkPropertyMetadata.BindsTwoWayByDefault標志——該標志指示屬性是使用單向綁定還是雙向綁定)。
通常,默認綁定模式也可正是期望的模式。然而,可設想一個示例,該例具有一個只讀的不允許用戶改變的文本框。對於這種情況,通過將模式設置為單向綁定可稍微降低一些開銷。
作為一條常用的經驗法則,明確設置綁定模式永遠不是壞主意。即使在文本框示例中,也值得通過包含Mode屬性來強調希望使用雙向綁定。
四、使用代碼創建綁定
在構建窗口時,在XAML標記中使用Binding標記擴展來聲明綁定表達式通常最高效。然而,也可使用代碼創建綁定。
下面的代碼演示了上面示例中顯示的TextBlock元素創建綁定:
Binding binding=new Binding(); binding.Source=sliderFontSize; binding.Path=new PropertyPath("Value"); binding.Mode=BindingMode.TwoWay; lblSampleText.SetBinding(TextBlock.FontSize,binding);
還可通過代碼使用BindingOperation類的兩個靜態方法移除綁定。ClearBinding()方法使用依賴項屬性(該屬性具有希望刪除的綁定)的引用作為參數,而ClearAllBindings()方法為元素刪除所有數據綁定:
BindingOperations.ClearAllBindings(lblSampleText);
ClearBinding()和ClearAllBindings()方法都使用ClearValue()方法,每個元素都從DependencyObject基類繼承了ClearValue()方法。ClearValue()方法簡單地移除屬性的本地值(對於這種情況,是數據綁定表達式)。
基於標記的綁定比通過代碼創建的綁定更常見,因為基於腳本的綁定更清晰並且需要完成的工作更少。一般使用標記創建它們的綁定,但在一些特殊情況下,會希望使用代碼創建綁定:
- 創建動態綁定:如果希望根據其他運行時信息修改綁定,或者根據環境創建不同的綁定,這時使用代碼創建綁定通常更合理(此外,也可在窗口的Resource集合中定義可能希望使用的每個綁定,並只添加是使用合適的綁定對象調用SetBinding()方法的代碼)。
- 刪除綁定:如果希望刪除綁定,從而可以通過普通方式設置屬性,需要借助ClearBinding()或ClearAllBindings()方法。僅為屬性應用新值是不夠的——如果正在使用雙向綁定,設置的值會傳播到鏈接的對象,並且兩個屬性保持同步。
- 創建自定義控件:為讓他人能更容易地修改你構建的自定義控件的外觀,需要將特定細節(如事件處理程序和數據綁定表達式)從標記移到代碼中。
五、使用代碼檢索綁定
可使用代碼檢索綁定並檢查其屬性,而不必考慮綁定最初是用代碼還是標記創建的。
可采用兩種方式來獲取綁定信息。第一種方式是使用靜態方法BindingOperations.GetBinding()來檢索相應的Binding對象。這需要提供兩個參數:綁定元素以及具有綁定表達式的屬性。
例如,如果具有如下綁定:
<TextBlock Margin="10" Name="lblSampleText" FontSize="{Binding ElementName=sliderFontSize, Path=Value}" Text="Simple Text"></TextBlock>
可使用如下代碼來獲取綁定:
Binding binding=BindingOperations.GetBinding(lblSampleText,TextBlock.FontSize);
一旦擁有綁定對象,就可以檢查其屬性。例如,綁定元素名Binding.ElementName提供了綁定表達式的值(這里是sliderFontSize)。Binding.Path提供的PropertyPath對象從綁定對象提取綁定值,Binding.Path.Path獲取綁定屬性的名稱(這里是Value)。還有Binding.Mode屬性,用於告知綁定合適更新目標元素。
如果必須在測試時添加診斷代碼,綁定對象會有趣一些。但WPF還允許通過調用BindingOperations.GetBindingExpression()方法獲得更實用的BindingExpression對象,該方法的參數與GetBinding()方法的參數相同:
BindingExpression expression = BindingOperations.GetBindingExpression(lblSampleText, TextBlock.FontSize);
BindingExpression對象包括一些屬性,用於復制Binding對象提供的信息。但迄今為止,最有趣的是ResolvedSource屬性,該屬性允許計算綁定表達式並獲得其結果——傳遞的本地數據。下面舉一個例子:
//Get the source element Slider boundObject=(Slider)expression.ResolvedSource; //Get any data you need from the source element,including it's bound property string boundData=boundObject.FontSize;
六、多綁定
上面的示例僅包含一個綁定,但如有必要,可設置TextBlock元素從文本框中獲取其文本,從單獨的顏色列表中選擇當前前景色和背景色等等,下面是一個示例:
<Window x:Class="DataBinding.MultipleBindings" xmlns="http://schemas.microsoft.com/winfx/2006/xaml/presentation" xmlns:x="http://schemas.microsoft.com/winfx/2006/xaml" Title="MultipleBindings" Height="300" Width="300"> <Grid Margin="5"> <Grid.RowDefinitions> <RowDefinition Height="Auto"></RowDefinition> <RowDefinition Height="Auto"></RowDefinition> <RowDefinition Height="Auto"></RowDefinition> <RowDefinition Height="Auto"></RowDefinition> <RowDefinition></RowDefinition> </Grid.RowDefinitions> <Slider Name="sliderFontSize" Margin="3" Minimum="1" Maximum="40" Value="10"></Slider> <TextBox Name="txtContent" Margin="3" Grid.Row="2">Sample Content</TextBox> <ListBox Margin="3" Grid.Row="3" Name="lstColors"> <ListBoxItem Tag="Blue">Blue</ListBoxItem> <ListBoxItem Tag="DarkBlue">Dark Blue</ListBoxItem> <ListBoxItem Tag="LightBlue">Light Blue</ListBoxItem> </ListBox> <TextBlock Margin="3" Name="lblSampleText" FontSize="{Binding ElementName=sliderFontSize, Path=Value}" Grid.Row="4" Text="{Binding ElementName=txtContent, Path=Text}" Foreground="{Binding ElementName=lstColors, Path=SelectedItem.Tag}" > </TextBlock> </Grid> </Window>
最終效果如下圖所示:

七、綁定更新
下面一個簡單的示例,當用戶在文本框中輸入字體大小時,發現文本字體大小並沒有立即變化,而是需要失去當前控件的焦點才會觸發。

會發生此問題的願意,是因為他們的行為由Binding.UpdateSourceTrigger屬性控制,該屬性可使用下表列出的某個值。當從文本框中取得文本並用於更新TextBlock.FontSize屬性時,看到的正式使用UpdateSourceTrigger.LostFocus方法從目標向源進行更新的例子:
表 UpdateSourceTrigger枚舉值

請記住,上表列出的值不印象目標的更新方式。他們僅控制TwoWay或OneWayToSource模式的綁定中源的更新方式。
根據上面介紹的內容,可改進文本框示例,從而當用戶在文本框中輸入內容時將變化應用於字體尺寸。方式如下:
<TextBox Name="txtBound" Text="{Binding ElementName=lblSampleText, Path=FontSize, UpdateSourceTrigger=PropertyChanged, Mode=TwoWay}" Width="100"></TextBox>
要完全控制源對象的更新時機,可選擇UpdateSourceTrigger.Explicit模式。如果在文本框示例中使用這種方法,當文本框失去焦點后不會發生任何事情。反而,由編寫代碼手動觸發更新。例如,可添加Apply按鈕,調用BindingExpression.UpdateSource()方法,觸發立即刷新行為並更新字體尺寸。
當然,在調用BindingExpression.UpdateSource()之前,需要一種方法來獲取BindingExpression對象。BindingExpression對象僅是將兩項內容封裝到一起的較小組裝包,這兩項內容是:已經學習過的Binding對象(通過BindingExpression.ParentBinding屬性提供)和由源綁定的對象(BindingExpression.DataItem)。此外,BindingExpression對象為觸發立即更新綁定的一部分提供了兩個方法:UpdateSource()和UdateTarget()方法。
為獲取BindingExpression對象,需要使用GetBindingExpression()方法,並傳入具有綁定的目標屬性,每個元素都從FrameworkElement基類繼承了該方法。下面的示例根據當前文本框中的文本改變TextBlock的字體大小:
BindingExpression binding=txtFontSize.GetBindingExpression(TextBox.TextProperty);
binding.UpdateSource();
八、綁定延遲
在極少數情況下,需要防止數據綁定觸發操作和修改源對象,至少在某一時刻是這樣的。例如,可能想在從文本框復制信息之前暫停,而不是在每次按鍵后獲取。或者,源對象在數據綁定屬性變化時執行處理器密集型操作。在此情況下,可能更添加短暫的延遲時間,避免過分頻繁地觸發操作。
在這些特殊情況下,可使用Binding對象Delay屬性。等到數毫秒,之后再提交更改。下面是文本框示例的修改版本,會在用戶停止輸入500毫秒后更新源對象:
<TextBox Name="txtBound" Text="{Binding ElementName=lblSampleText, Path=FontSize, UpdateSourceTrigger=PropertyChanged, Mode=TwoWay, Delay=500}" Width="100"></TextBox>
