一、什么是kvo?
key-value observing,觀察者模式
觀察者,觀察對象屬性的變化,當被觀察者該屬性發生變化時,觀察者會接收到通知,可以在回調函數中做相應的處理
二、有什么作用?
變化處理操作可以在同一個函數中進行,先前本人都會在每次修改屬性值的地方調用后續操作,比較繁瑣,修改的地方也比較多,現在只要在同一個函數中操作就可以
用kvo只要做監控就行,更加方便易用,減少代碼邏輯
三、使用場景:
當一個控件某個屬性變化需要做別的相應操作時,比較適合用kvo,只要當該屬性發生變化時,會發消息給觀察者,在回調函數中做相應的操作
四、實際例子:
一)解釋方法:
typedef NS_OPTIONS(NSUInteger, NSKeyValueObservingOptions) { NSKeyValueObservingOptionNew = 0x01,//改變后的值 NSKeyValueObservingOptionOld = 0x02,//改變前的值 NSKeyValueObservingOptionInitial NS_ENUM_AVAILABLE(10_5, 2_0) = 0x04, //addobserving之后會馬上調用observeValueForKeyPath,不會等到值改變 NSKeyValueObservingOptionPRior NS_ENUM_AVAILABLE(10_5, 2_0) = 0x08 //分2次調用。在值改變之前和值改變之后 };
NSKeyValueObservingOptionNew = 0x01,//改變后的值
NSKeyValueObservingOptionOld = 0x02,//改變前的值
這兩個用到的比較多
NSObject(NSKeyValueObserving) //一旦被觀察者屬性發生改變,就會調用此方法后續操作在這個方法中進行 - (void)observeValueForKeyPath:(NSString *)keyPath ofObject:(id)object change:(NSDictionary *)change context:(void *)context;
keyPath:是被觀察對象的屬性,字符串表示
object:被觀察對象
change:屬性改變的值,字典,通過 objectForKey (key為
FOUNDATION_EXPORT NSString *const NSKeyValueChangeKindKey;
FOUNDATION_EXPORT NSString *const NSKeyValueChangeNewKey;
FOUNDATION_EXPORT NSString *const NSKeyValueChangeOldKey;
FOUNDATION_EXPORT NSString *const NSKeyValueChangeIndexesKey;
FOUNDATION_EXPORT NSString *const NSKeyValueChangeNotificationIsPriorKey NS_AVAILABLE(10_5, 2_0);
對應addobserving指定的NSKeyValueObservingOptions
)
context:需要傳輸的數據(void *:任意指針類型),一般傳(__bridgevoid*)self 或者 nil,用戶也能傳別的
for example:
- (void)observeValueForKeyPath:(NSString *)keyPath ofObject:(id)object change:(NSDictionary *)change context:(void *)context { if (context == (__bridge void*)self) { if ([keyPath isEqualToString:kKeyPathForNavigationItemRightBarButtonItems]) { //取值 NSArray *rightBarButtonItems = [change objectForKey:NSKeyValueChangeNewKey]; //需要做操作 self.navigationItem.rightBarButtonItems = rightBarButtonItems; } } else { [super observeValueForKeyPath:keyPath ofObject:objectchange:changecontext:context]; } }
--------------------------------------------
二)接口方法
NSObject(NSKeyValueObserverRegistration) - (void)addObserver:(NSObject *)observer forKeyPath:(NSString *)keyPath options:(NSKeyValueObservingOptions)options context:(void *)context; - (void)removeObserver:(NSObject *)observer forKeyPath:(NSString *)keyPath context:(void *)context NS_AVAILABLE(10_7, 5_0); - (void)removeObserver:(NSObject *)observer forKeyPath:(NSString *)keyPath; NSArray(NSKeyValueObserverRegistration) - (void)addObserver:(NSObject *)observer toObjectsAtIndexes:(NSIndexSet *)indexes forKeyPath:(NSString *)keyPath options:(NSKeyValueObservingOptions)options context:(void *)context; - (void)removeObserver:(NSObject *)observer fromObjectsAtIndexes:(NSIndexSet *)indexes forKeyPath:(NSString *)keyPath context:(void *)contextNS_AVAILABLE(10_7,5_0); - (void)removeObserver:(NSObject *)observer fromObjectsAtIndexes:(NSIndexSet *)indexes forKeyPath:(NSString *)keyPath; - (void)addObserver:(NSObject *)observer forKeyPath:(NSString *)keyPath options:(NSKeyValueObservingOptions)options context:(void *)context; - (void)removeObserver:(NSObject *)observer forKeyPath:(NSString *)keyPath context:(void *)context NS_AVAILABLE(10_7, 5_0); - (void)removeObserver:(NSObject *)observer forKeyPath:(NSString *)keyPath; NSOrderedSet(NSKeyValueObserverRegistration) - (void)addObserver:(NSObject *)observer forKeyPath:(NSString *)keyPath options:(NSKeyValueObservingOptions)options context:(void *)context; - (void)removeObserver:(NSObject *)observer forKeyPath:(NSString *)keyPath context:(void *)context NS_AVAILABLE(10_7, 5_0); - (void)removeObserver:(NSObject *)observer forKeyPath:(NSString *)keyPath; NSSet(NSKeyValueObserverRegistration) - (void)addObserver:(NSObject *)observer forKeyPath:(NSString *)keyPath options:(NSKeyValueObservingOptions)options context:(void *)context; - (void)removeObserver:(NSObject *)observer forKeyPath:(NSString *)keyPath context:(void *)context NS_AVAILABLE(10_7, 5_0); - (void)removeObserver:(NSObject *)observer forKeyPath:(NSString *)keyPath; NSObject(NSKeyValueObserverNotification) //這些方法都為了手動通知用到 - (void)willChangeValueForKey:(NSString *)key; - (void)didChangeValueForKey:(NSString *)key; - (void)willChange:(NSKeyValueChange)changeKind valuesAtIndexes:(NSIndexSet *)indexes forKey:(NSString *)key; - (void)didChange:(NSKeyValueChange)changeKind valuesAtIndexes:(NSIndexSet *)indexes forKey:(NSString *)key; - (void)willChangeValueForKey:(NSString *)key withSetMutation:(NSKeyValueSetMutationKind)mutationKind usingObjects:(NSSet *)objects; - (void)didChangeValueForKey:(NSString *)key withSetMutation:(NSKeyValueSetMutationKind)mutationKind usingObjects:(NSSet *)objects; NSObject(NSKeyValueObservingCustomization) + (NSSet *)keyPathsForValuesAffectingValueForKey:(NSString *)keyNS_AVAILABLE(10_5,2_0); + (BOOL)automaticallyNotifiesObserversForKey:(NSString *)key;
** addObserver和removeObserver要成對出現
-----------------------------------------------------------------------
**手動通知:
有兩種通知觀察者的方式,自動通知和手動通知。顧名思義,手動通知需要在值變化時調用 willChangeValueForKey:和didChangeValueForKey: 方法通知調用者。為求簡便,我們一般使用自動通知。
要使用手動通知,需要在 automaticallyNotifiesObserversForKey方法中明確告訴cocoa,哪些鍵值要使用手動通知:
forExample:
[self willChangeValueForKey:@"frame"];self.frame = CGRectMake(0,0,320,100);[self didChangeValueForKey:@"frame"];
這時候就會調用
- (void)observeValueForKeyPath:(NSString *)keyPath ofObject:(id)object change:(NSDictionary *)change context:(void *)context;
//重新實現NSObject類中的automaticallyNotifiesObserversForKey:方法,返回yes表示自動通知。 + (BOOL)automaticallyNotifiesObserversForKey:(NSString*)key { //當這兩個值改變時,使用自動通知已注冊過的觀察者,觀察者需要實現observeValueForKeyPath:ofObject:change:context:方法 if ([key isEqualToString:@"frame"]) { return NO; } return [super automaticallyNotifiesObserversForKey:key]; } 這時候frame就必須要手動通知
*手動通知一般不用,為了方便,都自動通知,所以這部分知道就可以了
-----------------------------------------------------------------------
上面一些接口方法說明NSObject,NSArray,NSSet均實現了以上方法,因此我們不僅可以觀察普通對象,還可以觀察數組或結合類對象。
一般用的都是觀察NSObject的某個屬性
對NSArray進行觀察是觀察NSArray中每個model的屬性
NSSet和NSArray差不多,只不過NSSet是無序集合