KVO,即:Key-Value Observing,它提供一種機制,當指定的對象的屬性被修改後,則對象就會接受到通知。簡單的說就是每次指定的被觀察的對象的屬性被修改後,KVO就會自動通知相應的觀察者了。程序員
系統框架已經支持KVO,因此程序員在使用的時候很是簡單。app
1. 註冊,指定被觀察者的屬性,框架
2. 實現回調方法ide
3. 移除觀察函數
假設一個場景,股票的價格顯示在當前屏幕上,當股票價格更改的時候,實時顯示更新其價格。測試
1.定義DataModel,編碼
[cpp] view plaincopyspa
@interface StockData : NSObject { .net
NSString * stockName; orm
float price;
}
@end
@implementation StockData
@end
2.定義此model爲Controller的屬性,實例化它,監聽它的屬性,並顯示在當前的View裏邊
[cpp] view plaincopy
- (void)viewDidLoad
{
[super viewDidLoad];
stockForKVO = [[StockData alloc] init];
[stockForKVO setValue:@"searph" forKey:@"stockName"];
[stockForKVO setValue:@"10.0" forKey:@"price"];
[stockForKVO addObserver:self forKeyPath:@"price" options:NSKeyValueObservingOptionNew|NSKeyValueObservingOptionOld context:NULL];
myLabel = [[UILabel alloc]initWithFrame:CGRectMake(100, 100, 100, 30 )];
myLabel.textColor = [UIColor redColor];
myLabel.text = [stockForKVO valueForKey:@"price"];
[self.view addSubview:myLabel];
UIButton * b = [UIButton buttonWithType:UIButtonTypeRoundedRect];
b.frame = CGRectMake(0, 0, 100, 30);
[b addTarget:self action:@selector(buttonAction) forControlEvents:UIControlEventTouchUpInside];
[self.view addSubview:b];
}
3.當點擊button的時候,調用buttonAction方法,修改對象的屬性
[cpp] view plaincopy
-(void) buttonAction
{
[stockForKVO setValue:@"20.0" forKey:@"price"];
}
4. 實現回調方法
[cpp] view plaincopy
-(void)observeValueForKeyPath:(NSString *)keyPath ofObject:(id)object change:(NSDictionary *)change context:(void *)context
{
if([keyPath isEqualToString:@"price"])
{
myLabel.text = [stockForKVO valueForKey:@"price"];
}
}
5.增長觀察與取消觀察是成對出現的,因此須要在最後的時候,移除觀察者
[cpp] view plaincopy
- (void)dealloc
{
[super dealloc];
[stockForKVO removeObserver:self forKeyPath:@"price"];
[stockForKVO release];
}
KVC
KVC是KeyValueCoding的簡稱,它是一種能夠直接經過字符串的名字(key)來訪問類屬性的機制。而不是經過調用Setter、Getter方法訪問。
當使用KVO、Core Data、CocoaBindings、AppleScript(Mac支持)時,KVC是關鍵技術。
關鍵方法定義在:NSKeyValueCodingprotocol
KVC支持類對象和內建基本數據類型。
獲取值
valueForKey:,傳入NSString屬性的名字。
valueForKeyPath:,傳入NSString屬性的路徑,xx.xx形式。
valueForUndefinedKey它的默認實現是拋出異常,能夠重寫這個函數作錯誤處理。
修改值
setValue:forKey:
setValue:forKeyPath:
setValue:forUndefinedKey:
setNilValueForKey:當對非類對象屬性設置nil時,調用,默認拋出異常。
一對多關係成員的狀況
mutableArrayValueForKey:有序一對多關係成員 NSArray
mutableSetValueForKey:無序一對多關係成員 NSSet
1. 1 .Person類
2. @implementation Person
3. @synthesize name,age;//屬性name 將被監視
4. -(void) changeName
5. {
6. name=@"changeName directly";
7. }
8. @end
9.
10.
11. 2.PersonMonitor類 監視了name屬性
12. @implementation PersonMonitor
13.
14. - (void)observeValueForKeyPath:(NSString *)keyPath
15. ofObject:(id)object
16. change:(NSDictionary *)change
17. context:(void *)context
18. {
19. if ([keyPath isEqual:@"name"])
20. {
21. NSLog(@"change happen, old:%@ new:%@",[change objectForKey:NSKeyValueChangeOldKey],[change objectForKey:NSKeyValueChangeNewKey]);
22. }
23. }
24. @end
25.
26.
27. 3測試代碼
28.
29. //初始化被監視對象
30. Person *p =[[Person alloc] init];
31. //監視對象
32. PersonMonitor *pm= [[PersonMonitor alloc]init];
33. [p addObserver:pm forKeyPath:@"name" options:(NSKeyValueObservingOptionNew |NSKeyValueObservingOptionOld) context:nil];
34.
35. //測試前的數據
36. NSLog(@"p.name is %@",p.name);
37.
38. //經過setvalue 的方法,PersonMonitor的監視將被調用
39. [p setValue:@"name kvc" forKey:@"name"];
40.
41. //查看設置後的值
42. NSLog(@"p name get by kvc is %@",[p valueForKey:@"name"]);
43.
44. //效果和經過setValue 是一致的
45. p.name=@"name change by .name=";
46.
47. //經過person本身的函數來更改name
48. [p changeName];
49.
50. 結果是
51. 輸出
52. 2011-07-03 16:35:57.406 Cocoa[13970:903] p.name is name
53. 2011-07-03 16:35:57.418 Cocoa[13970:903] change happen, old:name new:name kvc
54. 2011-07-03 16:35:57.420 Cocoa[13970:903] p name get by kvc is name kvc
55. 2011-07-03 16:35:57.421 Cocoa[13970:903] change happen, old:name kvc new:name change by .name=
56. 最後一次修改是直接修改 因此無法產生通知