一、依賴屬性的全面解析
聽到依賴屬性,自然聯想到C#中屬性的概念。C#中屬性是抽象模型的核心部分,而依賴屬性是專門基于WPF建立的。在WPF庫實作中,依賴屬性使用普通的C#屬性進行了包裝,使得我們可以通過和以前一樣的方式來使用依賴屬性,但我們必須明确,在WPF中我們大多數都在使用依賴屬性,而不是使用屬性。依賴屬性重要性在于,在WPF核心特性,如動畫、資料綁定以及樣式中都需要使用到依賴屬性。既然WPF引入了依賴屬性,也自然有其引入的道理。WPF中的依賴屬性主要有以下三個優點:
- 依賴屬性加入了屬性變化通知、限制、驗證等功能。這樣可以使我們更友善地實作應用,同時大大減少了代碼量。許多之前需要寫很多代碼才能實作的功能,在WPF中可以輕松實作。
- 節約記憶體:在WinForm中,每個UI控件的屬性都賦予了初始值,這樣每個相同的控件在記憶體中都會儲存一份初始值。而WPF依賴屬性很好地解決了這個問題,它内部實作使用哈希表存儲機制,對多個相同控件的相同屬性的值都隻儲存一份。關于依賴屬性如何節約記憶體的更多内容參考:WPF的依賴屬性是怎麼節約記憶體的
- 支援多種提供對象:可以通過多種方式來設定依賴屬性的值。可以配合表達式、樣式和綁定來對依賴屬性設定值。
1.1 依賴屬性的定義
上面介紹了依賴屬性所帶來的好處,這時候,問題又來了,怎樣自己定義一個依賴屬性呢?C#屬性的定義大家再熟悉不過了。下面通過把C#屬性進行改寫成依賴屬性的方式來介紹依賴屬性的定義。下面是一個屬性的定義:
1 public class Person
2 {
3 public string Name { get; set; }
6 }
在把上面屬性改寫為依賴屬性之前,下面總結下定義依賴屬性的步驟:
- 讓依賴屬性的所在類型繼承自DependencyObject類。
- 使用public static 聲明一個DependencyProperty的變量,該變量就是真正的依賴屬性。
- 在類型的靜态構造函數中通過Register方法完成依賴屬性的中繼資料注冊。
- 提供一個依賴屬性的包裝屬性,通過這個屬性來完成對依賴屬性的讀寫操作。
根據上面的四個步驟,下面來把Name屬性來改寫成一個依賴屬性,具體的實作代碼如下所示:
![](https://img.laitimes.com/img/__Qf2AjLwojIjJCLyojI0JCLiYWan5SZk92Y5B3bj9CXzV2Zh1WavwVbvNmLzd2bsJmbj5ibv1WbvN2Lc9CX6MHc0RHaiojIsJye.gif)
// 1. 使類型繼承DependencyObject類
public class Person : DependencyObject
{
// 2. 聲明一個靜态隻讀的DependencyProperty 字段
public static readonly DependencyProperty nameProperty;
static Person()
{
// 3. 注冊定義的依賴屬性
nameProperty = DependencyProperty.Register("Name", typeof(string), typeof(Person),
new PropertyMetadata("Learning Hard",OnValueChanged));
}
// 4. 屬性包裝器,通過它來讀取和設定我們剛才注冊的依賴屬性
public string Name
{
get { return (string)GetValue(nameProperty); }
set { SetValue(nameProperty, value); }
}
private static void OnValueChanged(DependencyObject dpobj, DependencyPropertyChangedEventArgs e)
{
// 當隻發生改變時回調的方法
}
}
![](https://img.laitimes.com/img/__Qf2AjLwojIjJCLyojI0JCLiYWan5SZk92Y5B3bj9CXzV2Zh1WavwVbvNmLzd2bsJmbj5ibv1WbvN2Lc9CX6MHc0RHaiojIsJye.gif)
從上面代碼可以看出,依賴屬性是通過調用DependencyObject的GetValue和SetValue來對依賴屬性進行讀寫的。它使用哈希表來進行存儲的,對應的Key就是屬性的HashCode值,而值(Value)則是注冊的DependencyPropery;而C#中的屬性是類私有字段的封裝,可以通過對該字段進行操作來對屬性進行讀寫。總結為:屬性是字段的包裝,WPF中使用屬性對依賴屬性進行包裝。
1.2 依賴屬性的優先級
WPF允許在多個地方設定依賴屬性的值,則自然就涉及到依賴屬性擷取值的優先級問題。例如下面XMAL代碼,我們在三個地方設定了按鈕的背景顔色,那最終按鈕會讀取那個設定的值呢?是Green、Yellow還是Red?
![](https://img.laitimes.com/img/__Qf2AjLwojIjJCLyojI0JCLiYWan5SZk92Y5B3bj9CXzV2Zh1WavwVbvNmLzd2bsJmbj5ibv1WbvN2Lc9CX6MHc0RHaiojIsJye.gif)
<Window x:Class="DPSample.MainWindow"
xmlns="http://schemas.microsoft.com/winfx/2006/xaml/presentation"
xmlns:x="http://schemas.microsoft.com/winfx/2006/xaml"
Title="MainWindow" Height="350" Width="525">
<Grid>
<Button x:Name="myButton" Background="Green" Width="100" Height="30">
<Button.Style>
<Style TargetType="{x:Type Button}">
<Setter Property="Background" Value="Yellow"/>
<Style.Triggers>
<Trigger Property="IsMouseOver" Value="True">
<Setter Property="Background" Value="Red" />
</Trigger>
</Style.Triggers>
</Style>
</Button.Style>
Click Me
</Button>
</Grid>
</Window>
![](https://img.laitimes.com/img/__Qf2AjLwojIjJCLyojI0JCLiYWan5SZk92Y5B3bj9CXzV2Zh1WavwVbvNmLzd2bsJmbj5ibv1WbvN2Lc9CX6MHc0RHaiojIsJye.gif)
上面按鈕的背景顔色是Green。之是以背景色是Green,是因為WPF每通路一個依賴屬性,它都會按照下面的順序由高到底處理該值。具體優先級如下圖所示:
在上面XAML中,按鈕的本地值設定的是Green,自定義Style Trigger設定的為Red,自定義的Style Setter設定的為Yellow,由于這裡的本地值的優先級最高,是以按鈕的背景色或者的是Green值。如果此時把本地值Green去掉的話,此時按鈕的背景顔色是Yellow而不是Red。這裡盡管Style Trigger的優先級比Style Setter高,但是由于此時Style Trigger的IsMouseOver屬性為false,即滑鼠沒有移到按鈕上,一旦滑鼠移到按鈕上時,此時按鈕的顔色就為Red。此時才會展現出Style Trigger的優先級比Style Setter優先級高。是以上圖中優先級是比較理想情況下,很多時候還需要具體分析。
1.3 依賴屬性的繼承
依賴屬性是可以被繼承的,即父元素的相關設定會自動傳遞給所有的子元素。下面代碼示範了依賴屬性的繼承。
![](https://img.laitimes.com/img/__Qf2AjLwojIjJCLyojI0JCLiYWan5SZk92Y5B3bj9CXzV2Zh1WavwVbvNmLzd2bsJmbj5ibv1WbvN2Lc9CX6MHc0RHaiojIsJye.gif)
<Window x:Class="Custom_DPInherited.DPInherited"
xmlns="http://schemas.microsoft.com/winfx/2006/xaml/presentation"
xmlns:x="http://schemas.microsoft.com/winfx/2006/xaml"
xmlns:mc="http://schemas.openxmlformats.org/markup-compatibility/2006"
xmlns:d="http://schemas.microsoft.com/expression/blend/2008"
mc:Ignorable="d"
d:DesignHeight="300" d:DesignWidth="300"
FontSize="18"
Title="依賴屬性的繼承">
<StackPanel >
<Label Content="繼承自Window的FontSize" />
<Label Content="顯式設定FontSize"
TextElement.FontSize="36"/>
<StatusBar>Statusbar沒有繼承自Window的FontSize</StatusBar>
</StackPanel>
</Window>
![](https://img.laitimes.com/img/__Qf2AjLwojIjJCLyojI0JCLiYWan5SZk92Y5B3bj9CXzV2Zh1WavwVbvNmLzd2bsJmbj5ibv1WbvN2Lc9CX6MHc0RHaiojIsJye.gif)
上面的代碼的運作效果如下圖所示:
在上面XAML代碼中。Window.FontSize設定會影響所有内部子元素字型大小,這就是依賴屬性的繼承。如第一個Label沒有定義FontSize,是以它繼承了Window.FontSize值。但一旦子元素提供了顯式設定,這種繼承就會被打斷,是以Window.FontSize值對于第二個Label不再起作用。
這時,你可能已經發現了問題:StatusBar沒有顯式設定FontSize值,但它的字型大小沒有繼承Window.FontSize的值,而是保持了系統的預設值。那這是什麼原因呢?其實導緻這樣的問題:并不是所有元素都支援屬性值繼承的,如StatusBar、Tooptip和Menu控件。另外,StatusBar等控件截獲了從父元素繼承來的屬性,并且該屬性也不會影響StatusBar控件的子元素。例如,如果我們在StatusBar中添加一個Button。那麼這個Button的FontSize屬性也不會發生改變,其值為預設值。
前面介紹了依賴屬性的繼承,那我們如何把自定義的依賴屬性設定為可被其他控件繼承呢?通過AddOwer方法可以依賴屬性的繼承。具體的實作代碼如下所示:
![](https://img.laitimes.com/img/__Qf2AjLwojIjJCLyojI0JCLiYWan5SZk92Y5B3bj9CXzV2Zh1WavwVbvNmLzd2bsJmbj5ibv1WbvN2Lc9CX6MHc0RHaiojIsJye.gif)
1 public class CustomStackPanel : StackPanel
2 {
3 public static readonly DependencyProperty MinDateProperty;
4
5 static CustomStackPanel()
6 {
7 MinDateProperty = DependencyProperty.Register("MinDate", typeof(DateTime), typeof(CustomStackPanel), new FrameworkPropertyMetadata(DateTime.MinValue, FrameworkPropertyMetadataOptions.Inherits));
8 }
9
10 public DateTime MinDate
11 {
12 get { return (DateTime)GetValue(MinDateProperty); }
13 set { SetValue(MinDateProperty, value); }
14 }
15 }
16
17 public class CustomButton :Button
18 {
19 private static readonly DependencyProperty MinDateProperty;
20
21 static CustomButton()
22 {
23 // AddOwner方法指定依賴屬性的所有者,進而實作依賴屬性的繼承,即CustomStackPanel的MinDate屬性被CustomButton控件繼承。
24 // 注意FrameworkPropertyMetadataOptions的值為Inherits
25 MinDateProperty = CustomStackPanel.MinDateProperty.AddOwner(typeof(CustomButton), new FrameworkPropertyMetadata(DateTime.MinValue, FrameworkPropertyMetadataOptions.Inherits));
26 }
27
28 public DateTime MinDate
29 {
30 get { return (DateTime)GetValue(MinDateProperty); }
31 set { SetValue(MinDateProperty, value); }
32 }
33 }
![](https://img.laitimes.com/img/__Qf2AjLwojIjJCLyojI0JCLiYWan5SZk92Y5B3bj9CXzV2Zh1WavwVbvNmLzd2bsJmbj5ibv1WbvN2Lc9CX6MHc0RHaiojIsJye.gif)
接下來,你可以在XAML中進行測試使用,具體的XAML代碼如下:
![](https://img.laitimes.com/img/__Qf2AjLwojIjJCLyojI0JCLiYWan5SZk92Y5B3bj9CXzV2Zh1WavwVbvNmLzd2bsJmbj5ibv1WbvN2Lc9CX6MHc0RHaiojIsJye.gif)
<Window x:Class="Custom_DPInherited.MainWindow"
xmlns="http://schemas.microsoft.com/winfx/2006/xaml/presentation"
xmlns:x="http://schemas.microsoft.com/winfx/2006/xaml"
xmlns:local="clr-namespace:Custom_DPInherited"
xmlns:sys="clr-namespace:System;assembly=mscorlib"
Title="實作自定義依賴屬性的繼承" Height="350" Width="525">
<Grid>
<local:CustomStackPanel x:Name="customStackPanle" MinDate="{x:Static sys:DateTime.Now}">
<!--CustomStackPanel的依賴屬性-->
<ContentPresenter Content="{Binding Path=MinDate, ElementName=customStackPanle}"/>
<local:CustomButton Content="{Binding RelativeSource={x:Static RelativeSource.Self}, Path=MinDate}" Height="25"/>
</local:CustomStackPanel>
</Grid>
</Window>
![](https://img.laitimes.com/img/__Qf2AjLwojIjJCLyojI0JCLiYWan5SZk92Y5B3bj9CXzV2Zh1WavwVbvNmLzd2bsJmbj5ibv1WbvN2Lc9CX6MHc0RHaiojIsJye.gif)
上面XAML代碼中,顯示設定了CustomStackPanel的MinDate的值,而在CustomButton中卻沒有顯式設定其MinDate值。CustomButton的Content屬性的值是通過綁定MinDate屬性來進行擷取的,關于綁定的更多内容會在後面文章中分享。在這裡CustomButton中并沒有設定MinDate的值,但是CustomButton的Content的值卻是目前的時間,進而可以看出,此時CustomButton的MinDate屬性繼承了CustomStackPanel的MinDate的值,進而設定了其Content屬性。最終的效果如下圖所示:
1.4 隻讀依賴屬性
在C#屬性中,我們可以通過設定隻讀屬性來防止外界惡意更改該屬性值,同樣,在WPF中也可以設定隻讀依賴屬性。如IsMouseOver就是一個隻讀依賴屬性。那我們如何建立一個隻讀依賴屬性呢?其實隻讀的依賴屬性的定義方式與一般依賴屬性的定義方式基本一樣。隻讀依賴屬性僅僅是用DependencyProperty.RegisterReadonly替換了DependencyProperty.Register而已。下面代碼實作了一個隻讀依賴屬性。
![](https://img.laitimes.com/img/__Qf2AjLwojIjJCLyojI0JCLiYWan5SZk92Y5B3bj9CXzV2Zh1WavwVbvNmLzd2bsJmbj5ibv1WbvN2Lc9CX6MHc0RHaiojIsJye.gif)
1 public partial class MainWindow : Window
2 {
3 public MainWindow()
4 {
5 InitializeComponent();
6
7 // 内部使用SetValue來設定值
8 SetValue(counterKey, 8);
9 }
10
11 // 屬性包裝器,隻提供GetValue,你也可以設定一個private的SetValue進行限制。
12 public int Counter
13 {
14 get { return (int)GetValue(counterKey.DependencyProperty); }
15 }
16
17 // 使用RegisterReadOnly來代替Register來注冊一個隻讀的依賴屬性
18 private static readonly DependencyPropertyKey counterKey =
19 DependencyProperty.RegisterReadOnly("Counter",
20 typeof(int),
21 typeof(MainWindow),
22 new PropertyMetadata(0));
23 }
![](https://img.laitimes.com/img/__Qf2AjLwojIjJCLyojI0JCLiYWan5SZk92Y5B3bj9CXzV2Zh1WavwVbvNmLzd2bsJmbj5ibv1WbvN2Lc9CX6MHc0RHaiojIsJye.gif)
對應的XAML代碼為:
![](https://img.laitimes.com/img/__Qf2AjLwojIjJCLyojI0JCLiYWan5SZk92Y5B3bj9CXzV2Zh1WavwVbvNmLzd2bsJmbj5ibv1WbvN2Lc9CX6MHc0RHaiojIsJye.gif)
<Window x:Class="ReadOnlyDP.MainWindow"
Name="ThisWin"
xmlns="http://schemas.microsoft.com/winfx/2006/xaml/presentation"
xmlns:x="http://schemas.microsoft.com/winfx/2006/xaml"
Title="ReadOnly Dependency Property" Height="350" Width="525">
<Grid>
<Viewbox>
<TextBlock Text="{Binding ElementName=ThisWin, Path=Counter}"/>
</Viewbox>
</Grid>
</Window>
![](https://img.laitimes.com/img/__Qf2AjLwojIjJCLyojI0JCLiYWan5SZk92Y5B3bj9CXzV2Zh1WavwVbvNmLzd2bsJmbj5ibv1WbvN2Lc9CX6MHc0RHaiojIsJye.gif)
此時Counter包裝的counterKey就是一個隻讀依賴屬性,因為其定義為private的,是以在類外也不能使用DependencyObject.SetValue方法來對其值,而包裝的Counter屬性又隻提供了GetValue方法,是以類外部隻能對該依賴屬性進行讀取,而不能對其指派。此時運作效果如下圖所示。
1.5 附加屬性
WPF中還有一類特殊的屬性——附加屬性。附加是一種特殊的依賴屬性。它允許給一個對象添加一個值,而該對象可能對這個值一無所知。附加屬性最常見的例子就是布局容器中DockPanel類中的Dock附加屬性和Grid類中Row和Column附加屬性。那問題又來了,我們怎樣在自己的類中定義一個附加屬性呢?其實定義附加屬性和定義一般的依賴屬性一樣沒什麼差別,隻是用RegisterAttached方法代替了Register方法罷了。下面代碼示範了附加屬性的定義。
![](https://img.laitimes.com/img/__Qf2AjLwojIjJCLyojI0JCLiYWan5SZk92Y5B3bj9CXzV2Zh1WavwVbvNmLzd2bsJmbj5ibv1WbvN2Lc9CX6MHc0RHaiojIsJye.gif)
public class AttachedPropertyClass
{
// 通過使用RegisterAttached來注冊一個附加屬性
public static readonly DependencyProperty IsAttachedProperty =
DependencyProperty.RegisterAttached("IsAttached", typeof(bool), typeof(AttachedPropertyClass),
new FrameworkPropertyMetadata((bool)false));
// 通過靜态方法的形式暴露讀的操作
public static bool GetIsAttached(DependencyObject dpo)
{
return (bool)dpo.GetValue(IsAttachedProperty);
}
public static void SetIsAttached(DependencyObject dpo, bool value)
{
dpo.SetValue(IsAttachedProperty, value);
}
}
![](https://img.laitimes.com/img/__Qf2AjLwojIjJCLyojI0JCLiYWan5SZk92Y5B3bj9CXzV2Zh1WavwVbvNmLzd2bsJmbj5ibv1WbvN2Lc9CX6MHc0RHaiojIsJye.gif)
在上面代碼中,IsAttached就是一個附加屬性,附加屬性沒有采用CLR屬性進行封裝,而是使用靜态SetIsAttached方法和GetIsAttached方法來存取IsAttached值。這兩個靜态方法内部一樣是調用SetValue和GetValue來對附加屬性讀寫的。
1.6 依賴屬性驗證和強制
在定義任何類型的屬性時,都需要考慮錯誤設定屬性的可能性。對于傳統的CLR屬性,可以在屬性的設定器中進行屬性值的驗證,不滿足條件的值可以抛出異常。但對于依賴屬性來說,這種方法不合适,因為依賴屬性通過SetValue方法來直接設定其值的。然而WPF有其代替的方式,WPF中提供了兩種方法來用于驗證依賴屬性的值。
- ValidateValueCallback:該回調函數可以接受或拒絕新值。該值可作為DependencyProperty.Register方法的一個參數。
- CoerceValueCallback:該回調函數可将新值強制修改為可被接受的值。例如某個依賴屬性Age的值範圍是0到120,在該回調函數中,可以對設定的值進行強制修改,對于不滿足條件的值,強制修改為滿足條件的值。如當設定為負值時,可強制修改為0。該回調函數可作為PropertyMetadata構造函數參數進行傳遞。
當應用程式設定一個依賴屬性時,所涉及的驗證過程如下所示:
- 首先,CoerceValueCallback方法可以修改提供的值或傳回DependencyProperty.UnsetValue。
- 如果CoerceValueCallback方法強制修改了提供的值,此時會激活ValidateValueCallback方法進行驗證,如果該方法傳回為true,表示該值合法,被認為可被接受的,否則拒絕該值。不像CoerceValueCallback方法,ValidateValueCallback方法不能通路設定屬性的實際對象,這意味着你不能檢查其他屬性值。即該方法中不能對類的其他屬性值進行通路。
- 如果上面兩個階段都成功的話,最後會觸發PropertyChangedCallback方法來觸發依賴屬性值的更改。
下面代碼示範了基本的流程。
![](https://img.laitimes.com/img/__Qf2AjLwojIjJCLyojI0JCLiYWan5SZk92Y5B3bj9CXzV2Zh1WavwVbvNmLzd2bsJmbj5ibv1WbvN2Lc9CX6MHc0RHaiojIsJye.gif)
1 class Program
2 {
3 static void Main(string[] args)
4 {
5 SimpleDPClass sDPClass = new SimpleDPClass();
6 sDPClass.SimpleDP = 2;
7 Console.ReadLine();
8 }
9 }
10
11 public class SimpleDPClass : DependencyObject
12 {
13 public static readonly DependencyProperty SimpleDPProperty =
14 DependencyProperty.Register("SimpleDP", typeof(double), typeof(SimpleDPClass),
15 new FrameworkPropertyMetadata((double)0.0,
16 FrameworkPropertyMetadataOptions.None,
17 new PropertyChangedCallback(OnValueChanged),
18 new CoerceValueCallback(CoerceValue)),
19 new ValidateValueCallback(IsValidValue));
20
21 public double SimpleDP
22 {
23 get { return (double)GetValue(SimpleDPProperty); }
24 set { SetValue(SimpleDPProperty, value); }
25 }
26
27 private static void OnValueChanged(DependencyObject d, DependencyPropertyChangedEventArgs e)
28 {
29 Console.WriteLine("當值改變時,我們可以做的一些操作,具體可以在這裡定義: {0}", e.NewValue);
30 }
31
32 private static object CoerceValue(DependencyObject d, object value)
33 {
34 Console.WriteLine("對值進行限定,強制值: {0}", value);
35 return value;
36 }
37
38 private static bool IsValidValue(object value)
39 {
40 Console.WriteLine("驗證值是否通過,傳回bool值,如果傳回True表示驗證通過,否則會以異常的形式暴露: {0}", value);
41 return true;
42 }
43 }
![](https://img.laitimes.com/img/__Qf2AjLwojIjJCLyojI0JCLiYWan5SZk92Y5B3bj9CXzV2Zh1WavwVbvNmLzd2bsJmbj5ibv1WbvN2Lc9CX6MHc0RHaiojIsJye.gif)
其運作結果如下圖所示:
從運作結果可以看出,此時并沒有按照上面的流程先Coerce後Validate的順序執行,這可能是WPF内部做了一些特殊的處理。當屬性被改變時,首先會調用Validate來判斷傳入的value是否有效,如果無效就不繼續後續操作。并且CoerceValue後面并沒有運作ValidateValue,而是直接調用PropertyChanged。這是因為CoerceValue操作并沒有強制改變屬性的值,而前面對這個值已經驗證過了,是以也就沒有必要再運作Valudate方法來進行驗證了。但是如果在Coerce中改變了Value的值,那麼還會再次調用Valudate操作來驗證值是否合法。
1.7 依賴屬性的監聽
我們可以用兩種方法對依賴屬性的改變進行監聽。這兩種方法是:
- 使用DependencyPropertyDescriptor類
- 使用OverrideMetadata的方式。
下面分别使用這兩種方式來實作下對依賴屬性的監聽。
第一種方式:定義一個派生于依賴屬性所在的類,然後重寫依賴屬性的中繼資料并傳遞一個PropertyChangedCallback參數即可,具體的實作如下代碼所示:
![](https://img.laitimes.com/img/__Qf2AjLwojIjJCLyojI0JCLiYWan5SZk92Y5B3bj9CXzV2Zh1WavwVbvNmLzd2bsJmbj5ibv1WbvN2Lc9CX6MHc0RHaiojIsJye.gif)
1 public class MyTextBox : TextBox
2 {
3 public MyTextBox()
4 : base()
5 {
6 }
7
8 static MyTextBox()
9 {
10 //第一種方法,通過OverrideMetadata
11 TextProperty.OverrideMetadata(typeof(MyTextBox), new FrameworkPropertyMetadata(new PropertyChangedCallback(TextPropertyChanged)));
12 }
13
14 private static void TextPropertyChanged(DependencyObject sender, DependencyPropertyChangedEventArgs args)
15 {
16 MessageBox.Show("", "Changed");
17 }
18 }
![](https://img.laitimes.com/img/__Qf2AjLwojIjJCLyojI0JCLiYWan5SZk92Y5B3bj9CXzV2Zh1WavwVbvNmLzd2bsJmbj5ibv1WbvN2Lc9CX6MHc0RHaiojIsJye.gif)
第二種方法:這個方法更加簡單,擷取DependencyPropertyDescriptor并調用AddValueChange方法為其綁定一個回調函數。具體實作代碼如下所示:
![](https://img.laitimes.com/img/__Qf2AjLwojIjJCLyojI0JCLiYWan5SZk92Y5B3bj9CXzV2Zh1WavwVbvNmLzd2bsJmbj5ibv1WbvN2Lc9CX6MHc0RHaiojIsJye.gif)
public MainWindow()
{
InitializeComponent();
//第二種方法,通過OverrideMetadata
DependencyPropertyDescriptor descriptor = DependencyPropertyDescriptor.FromProperty(TextBox.TextProperty, typeof(TextBox));
descriptor.AddValueChanged(tbxEditMe, tbxEditMe_TextChanged);
}
private void tbxEditMe_TextChanged(object sender, EventArgs e)
{
MessageBox.Show("", "Changed");
}
![](https://img.laitimes.com/img/__Qf2AjLwojIjJCLyojI0JCLiYWan5SZk92Y5B3bj9CXzV2Zh1WavwVbvNmLzd2bsJmbj5ibv1WbvN2Lc9CX6MHc0RHaiojIsJye.gif)
二、總結
到這裡,依賴屬性的介紹就結束了。WPF中的依賴屬性通過一個靜态隻讀字段進行定義,并且在靜态構造函數中進行注冊,最後通過.NET傳統屬性進行包裝,使其使用與傳統的.NET屬性并無兩樣。在後面一篇文章将分享WPF中新的事件機制——路由事件。
(轉自:https://www.cnblogs.com/zhili/p/WPFDependencyProperty.html)