IOS開發-KVO

來源:互聯網
上載者:User

標籤:style   blog   color   io   os   ar   使用   for   strong   

一、什麼是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是無序集合

IOS開發-KVO

聯繫我們

該頁面正文內容均來源於網絡整理,並不代表阿里雲官方的觀點,該頁面所提到的產品和服務也與阿里云無關,如果該頁面內容對您造成了困擾,歡迎寫郵件給我們,收到郵件我們將在5個工作日內處理。

如果您發現本社區中有涉嫌抄襲的內容,歡迎發送郵件至: info-contact@alibabacloud.com 進行舉報並提供相關證據,工作人員會在 5 個工作天內聯絡您,一經查實,本站將立刻刪除涉嫌侵權內容。

A Free Trial That Lets You Build Big!

Start building with 50+ products and up to 12 months usage for Elastic Compute Service

  • Sales Support

    1 on 1 presale consultation

  • After-Sales Support

    24/7 Technical Support 6 Free Tickets per Quarter Faster Response

  • Alibaba Cloud offers highly flexible support services tailored to meet your exact needs.