INotifyPropertyChanged接口的实现

何时实现INotifyPropertyChanged接口

官方解释:INotifyPropertyChanged  接口用于向客户端(通常是执行绑定的客户端)发出某一属性值已更改的通知。官方解释的很模糊,估计是个人看了都不知道到底什么时候需要实现INotifyPropertyChanged接口.小梦通过实际测试给出明确结论:

首先:OneTime模式:毫无意义,因为它的绑定只有初始时候绑定一次,根本谈不上改变!自然也就谈不上实现INotifyPropertyChanged接口.

然后是OneWay模式:我们知道OneWay模式的含义是:绑定源的每一次变化都会通知绑定目标,但是绑定目标的改变不会改变绑定源.当绑定源的数据实体类没有实现INotifyPropertyChanged接口时,当我们改变了数据源,我们会发现绑定目标的UI上的相应的数据不会立即变化.所以这时候就需要我们来实现INotifyPropertyChanged接口.

最后是TwoWay模式:在TwoWay模式下,当绑定源的数据实体类没有实现INotifyPropertyChanged接口时,我们发现.控件的更改会让数据源立即发改变,但是改变数据源,绑定目标控件却不会立即发生改变!所以当我们需要数据源改变时相对应的UI立即改变时,就需要实现INotifyPropertyChanged接口.

总之:就是当数据源改变并需要UI立即改变时我们需要实现INotifyPropertyChanged接口.

我们可以通过这个示例来明确的体会这一点:

C#

<StackPanel>

<TextBox Header="编号" Text="{Binding ID,Mode=OneTime}" Name="tbxID" ></TextBox>

<TextBox Header="书名" Text="{Binding Title,Mode=OneWay}" Name="tbxTitle" ></TextBox>

<TextBox Header="价格" Text="{Binding Price,Mode=TwoWay}" Name="tbxPrice" ></TextBox>

<Button Content="通过数据源修改控件的值" Click="Button_Click"></Button>

<Button Content="直接修改控件的值" Click="Button_Click_1" />

<Button Content="通过控件修改数据源的值" Click="Button_Click_2" />

</StackPanel>

1

2

3

4

5

6

7

8

9

10

11

12

13

14

15

<StackPanel>

<TextBox  Header="编号" Text="{Binding ID,Mode=OneTime}" Name="tbxID"  ></TextBox>

<TextBox Header="书名" Text="{Binding Title,Mode=OneWay}" Name="tbxTitle" ></TextBox>

<TextBox  Header="价格" Text="{Binding Price,Mode=TwoWay}" Name="tbxPrice" ></TextBox>

<Button Content="通过数据源修改控件的值"  Click="Button_Click"></Button>

<Button Content="直接修改控件的值"     Click="Button_Click_1" />

<Button Content="通过控件修改数据源的值"   Click="Button_Click_2" />

</StackPanel>

后台代码:

C#

namespace INotifyPropertyChangedDEMO
{
/// <summary>
/// 可用于自身或导航至 Frame 内部的空白页。
/// </summary>

public sealed partial class MainPage : Page
{
Book book = new Book();
public MainPage()
{
this.InitializeComponent();

this.NavigationCacheMode = NavigationCacheMode.Required;
book.ID = 0;
book.Title = "ASP.NET 开发手册";
book.Price = 40;
st.DataContext = book;
}
private void Button_Click(object sender, RoutedEventArgs e)//通过修改数据源修改控件的值
{
book.ID = 100;
book.Price = 50;
book.Title = "SL开发手册";
}

private async void Button_Click_1(object sender, RoutedEventArgs e)//显示数据源的值
{
await new MessageDialog(book.ID.ToString() + " " + book.Title.ToString() + " " + book.Price.ToString()).ShowAsync();
}

public class Book : INotifyPropertyChanged
//INotifyPropertChanged 接口定义了一个当属性值更改时执行的事件,事件名称为PropertyChanged。
//这个是在继承这个接口的类必须要实现的事件

{
private int _id;
public int ID
{
get { return _id; }
set
{
_id = value;
//NotifyPropertyChange("ID");
}
}
private string _title;
public string Title
{
get { return _title; }
set
{
_title = value;
//NotifyPropertyChange("Title");
}
}
private double _price;
public double Price
{
get { return _price; }
set
{
_price = value;
//NotifyPropertyChange("Price");
}
}
public event PropertyChangedEventHandler PropertyChanged;
//PropertyChangedEventArgs类型,这个类用于传递更改值的属性的名称,实现向客户端已经更改的属性发送更改通知。属性的名称为字符串类型。
private void NotifyPropertyChange(string propertyName)
{
if (PropertyChanged != null)
{
//根据PropertyChanged事件的委托类,实现PropertyChanged事件:
PropertyChanged(this, new PropertyChangedEventArgs(propertyName));
}
}
}
}
}

1

2

3

4

5

6

7

8

9

10

11

12

13

14

15

16

17

18

19

20

21

22

23

24

25

26

27

28

29

30

31

32

33

34

35

36

37

38

39

40

41

42

43

44

45

46

47

48

49

50

51

52

53

54

55

56

57

58

59

60

61

62

63

64

65

66

67

68

69

70

71

72

73

74

75

76

77

78

79

namespace INotifyPropertyChangedDEMO

{

/// <summary>

/// 可用于自身或导航至 Frame 内部的空白页。

/// </summary>

public sealed partial class MainPage : Page

{

Book book = new Book();

public MainPage()

{

this.InitializeComponent();

this.NavigationCacheMode = NavigationCacheMode.Required;

book.ID = 0;

book.Title = "ASP.NET 开发手册";

book.Price = 40;

st.DataContext = book;

}

private void Button_Click(object sender, RoutedEventArgs e)//通过修改数据源修改控件的值

{

book.ID = 100;

book.Price = 50;

book.Title = "SL开发手册";

}

private async void Button_Click_1(object sender, RoutedEventArgs e)//显示数据源的值

{

await new MessageDialog(book.ID.ToString() + " " + book.Title.ToString() + " " + book.Price.ToString()).ShowAsync();

}

public class Book : INotifyPropertyChanged

//INotifyPropertChanged 接口定义了一个当属性值更改时执行的事件,事件名称为PropertyChanged。

//这个是在继承这个接口的类必须要实现的事件

{

private int _id;

public int ID

{

get { return _id; }

set

{

_id = value;

//NotifyPropertyChange("ID");

}

}

private string _title;

public string Title

{

get { return _title; }

set

{

_title = value;

//NotifyPropertyChange("Title");

}

}

private double _price;

public double Price

{

get { return _price; }

set

{

_price = value;

//NotifyPropertyChange("Price");

}

}

public event PropertyChangedEventHandler PropertyChanged;

//PropertyChangedEventArgs类型,这个类用于传递更改值的属性的名称,实现向客户端已经更改的属性发送更改通知。属性的名称为字符串类型。

private void NotifyPropertyChange(string propertyName)

{

if (PropertyChanged != null)

{

//根据PropertyChanged事件的委托类,实现PropertyChanged事件:

PropertyChanged(this, new PropertyChangedEventArgs(propertyName));

}

}

}

}

}

大家运行这个示例可以明显体会INotifyPropertyChanged接口的作用.

如何实现INotifyPropertyChanged接口

上面示例的INotifyPropertyChanged接口的实现方式是最常见和最普遍的.

我们可以利用CallerMemberNameAttribute特性来简化一下,这个特性可以根据调用方来决定传入哪个属性的名字.:

C#

protected void OnPropertyChanged([CallerMemberName] string propertyName = null)
{
var eventHandler = this.PropertyChanged;
if (eventHandler != null)
eventHandler(this, new PropertyChangedEventArgs(propertyName));
}

1

2

3

4

5

6

protected void OnPropertyChanged([CallerMemberName] string propertyName = null)

{

var eventHandler = this.PropertyChanged;

if (eventHandler != null)

eventHandler(this, new PropertyChangedEventArgs(propertyName));

}

这样我们在调用时可以这样调用:

NotifyPropertyChange(“ID”)  改为:OnPropertyChanged();

INotifyPropertyChanged接口的最佳实现方式:

这个所谓的最佳实现方式 是channel 9的视频中说的,实现方式如下:

C#

public class ModelBase : INotifyPropertyChanged
{
public event PropertyChangedEventHandler PropertyChanged;
protected bool SetProperty<T>(ref T storage, T value, [CallerMemberName] String propertyName = null)
{
if (object.Equals(storage, value)) return false;
storage = value;
this.OnPropertyChanged(propertyName);
return true;
}

protected void OnPropertyChanged([CallerMemberName] string propertyName = null)
{
var eventHandler = this.PropertyChanged;
if (eventHandler != null)
eventHandler(this, new PropertyChangedEventArgs(propertyName));
}
}

1

2

3

4

5

6

7

8

9

10

11

12

13

14

15

16

17

18

public class ModelBase : INotifyPropertyChanged

{

public event PropertyChangedEventHandler PropertyChanged;

protected bool SetProperty<T>(ref T storage, T value, [CallerMemberName] String propertyName = null)

{

if (object.Equals(storage, value)) return false;

storage = value;

this.OnPropertyChanged(propertyName);

return true;

}

protected void OnPropertyChanged([CallerMemberName] string propertyName = null)

{

var eventHandler = this.PropertyChanged;

if (eventHandler != null)

eventHandler(this, new PropertyChangedEventArgs(propertyName));

}

}

相应的调用方式进一步简化:

C#

private string name;

public string Name
{
get { return name; }
set
{ this.SetProperty(ref this.name, value); }
}

1

2

3

4

5

6

7

8

private string name;

public string Name

{

get { return name; }

set

{ this.SetProperty(ref this.name, value); }

}

时间: 2024-10-25 11:39:40

INotifyPropertyChanged接口的实现的相关文章

INotifyPropertyChanged 接口

2016-07-24 INotifyPropertyChanged 接口 用于向客户端(通常是执行绑定的客户端)发出某一属性值已更改的通知. 例如,考虑一个带有名为 FirstName 属性的 Person 对象.若要提供一般性属性更改通知,则 Person 类型实现 INotifyPropertyChanged 接口并在 FirstName 更改时引发 PropertyChanged 事件. 若要在将客户端与数据源进行绑定时发出更改通知,则绑定类型应具有下列任一功能: 实现 INotifyPr

INotifyPropertyChanged 接口 CallerMemberName属性

2016-07-24 调用方信息 使用调用方信息属性,可以获取关于调用方的信息传递给方法. 可以获取源代码.行号在源代码和调用方的成员名称的文件路径. 此信息用于跟踪,调试和创建诊断工具非常有用. 若要获取此信息,则使用适用于可选参数,每个都有一个默认的属性. 下表列出了 System.Runtime.CompilerServices 命名空间中定义的调用方信息属性: 特性 说明 类型 CallerFilePathAttribute 包含调用方源文件的完整路径. 这是文件路径在编译时. Stri

转载:WPF MVVM之INotifyPropertyChanged接口的几种实现方式

原文地址:http://www.cnblogs.com/xiwang/ 序言 借助WPF/Sliverlight强大的数据绑定功能,可以比实现比MFC,WinForm更加优雅轻松的数据绑定.但是在使用WPF/Silverlight绑定时,有件事情是很苦恼的:当ViewModel对象放生改变,需要通知UI.我们可以让VM对象实现INotifyPropertyChanged接口,通过事件来通知UI.但问题就出现这里…… 一,描述问题 情形:现在需要将一个Person对象的Name熟悉双向绑定到UI中

WPF系列之三:实现类型安全的INotifyPropertyChanged接口,可以不用“Magic string” 么?

通常实现INotifyPropertyChanged接口很简单,为你的类只实现一个PropertyChanged 的Event就可以了. 例如实现一个简单的ViewModel1类: public class ViewModel1 : INotifyPropertyChanged { private string _data; public string Data { get { return _data; } set { if (_data == value) return; _data = v

INotifyPropertyChanged接口的详细说明

在windows phone开发8.1:数据绑定中,我们了解了数据绑定的基本知识.今后几篇文章会继续深入了解数据绑定.今天我们来看在数据绑定中十分重要的INotifyPropertyChanged接口的实现. 何时实现INotifyPropertyChanged接口 官方解释:INotifyPropertyChanged  接口用于向客户端(通常是执行绑定的客户端)发出某一属性值已更改的通知.官方解释的很模糊,估计是个人看了都不知道到底什么时候需要实现INotifyPropertyChanged

【.NET深呼吸】INotifyPropertyChanged接口的真故事

无论是在流氓腾的问问社区,还是在黑度贴吧,或是“厕所等你”论坛上,曾经看到过不少朋友讨论INotifyPropertyChanged接口.不少朋友认为该接口是为双向绑定而使用的,那么,真实的情况是这样的吗? INotifyPropertyChanged接口位于System.ComponentModel命名空间,在该命名空间下还有另一个接口:INotifyPropertyChanging.INotifyPropertyChanging接口定义了PropertyChanging事件,应该在在属性值正

INotifyPropertyChanged接口的PropertyChanged 事件

INotifyPropertyChanged 接口用于向客户端(通常是执行绑定的客户端)发出某一属性值已更改的通知. 例如,考虑一个带有名为 FirstName 属性的 Person 对象. 若要提供一般性属性更改通知,则 Person 类型实现 INotifyPropertyChanged 接口并在 FirstName更改时引发 PropertyChanged 事件. 若要在将客户端与数据源进行绑定时发出更改通知,则绑定类型应具有下列任一功能: 实现 INotifyPropertyChange

MVVM设计模式基础知识--INotifyPropertyChanged接口

在.NET平台上,数据绑定是一项令人十分愉快的技术.利用数据绑定能减少代码,简化控制逻辑. 通常,可以将某个对象的一个属性绑定到一个可视化的控件上,当属性值改变时,控件上的显示数据也随之发生变化.要实现这一功能,只需要为自定义对象实现 INotifyPropertyChanged 接口即可.此接口中定义了 PropertyChanged 事件,我们只需在属性值改变时触发该事件即可. INotifyPropertyChanged 接口是 WPF/Silverlight 开发中非常重要的接口, 它构

[译]WPF MVVM 架构 Step By Step(5)(添加actions和INotifyPropertyChanged接口)

原文:[译]WPF MVVM 架构 Step By Step(5)(添加actions和INotifyPropertyChanged接口) 应用不只是包含textboxs和labels,还包含actions,如按钮和鼠标事件等.接下来我们加上一些像按钮这样的UI元素来看MVVM类怎么演变的.与之前的UI相比,这次我们加上一个"Cal Tax"按钮,当我们点击这个依赖于“sales amount”的按钮时,它会计算税费并显示在同窗口内. 为了完成所述的功能,我们先在Model类中添加一个