一,概述
KVO,即:Key-Value Observing,它提供一種機制,當指定的對象的屬性被修改後,則對象就會接受到通知。簡單的說就是每次指定的被觀察的對象的屬性被修改後,KVO就會自動通知相應的觀察者了。
二,使用方法
系統框架已經支持KVO,所以程序員在使用的時候非常簡單。
1. 注冊,指定被觀察者的屬性,
2. 實現回調方法
3. 移除觀察
三,實例:
假設一個場景,股票的價格顯示在當前屏幕上,當股票價格更改的時候,實時顯示更新其價格。
1.定義DataModel,
[cpp] view plaincopy
- @interface StockData : NSObject {
- NSString * stockName;
- 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];
- }
四,小結
KVO這種編碼方式使用起來很簡單,很適用與datamodel修改後,引發的UIVIew的變化這種情況,就像上邊的例子那樣,當更改屬性的值後,監聽對象會立即得到通知。
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. 最後一次修改是直接修改 所以沒法產生通知
四,小結
KVO/KVC這種編碼方式使用起來很簡單,很適用與datamodel修改後,引發的UIVIew的變化這種情況,就像上邊的例子那樣,當更改屬性的值後,監聽對象會立即得到通知。