图文课程 职群旗下

  1. 首页
  2. 所有分类
  3. IOS
  4. OC 观察者模式(通知中心,KVO)

OC 观察者模式(通知中心,KVO)

OC 观察者模式(通知中心,KVO)


什么是观察者模式???

A对B的变化感兴趣,就注册为B的观察者,当B发生变化时通知A,告知B发生了变化。这就是观察者模式。
观察者模式定义了一种一对多的依赖关系,让多个观察者对象同时监听某一个主题对象。这个主题对象在状态上发生变化时,会通知所有观察者对象,使它们能够自动更新自己或者做出相应的一些动作。在IOS开发中我们接触到观察者模式的实现方式有NSNotificationCenter、KVO等.

通知中心

以NSNotificationCenter为中心,观察者往Center中注册对某个主题对象的变化感兴趣,主题对象通过NSNotificationCenter进行变化广播。所有的观察和监听行为都向同一个中心注册,所有对象的变化也都通过同一个中心向外广播。

通知中心执行顺序

  1. 通过NSNotificationCenter的addObserver:selector:name:object接口来注册通知中心.
  2. 被观察的对象,通过postNotificationName:object:userInfo:向通知中心发送某一类型通知.
  3. 当有通知来的时候,Center会调用观察者注册的接口来广播通知,同时传递存储着更改内容的NSNotification对象.
  4. 当该通知不再使用时,可以在dealloc方法里removeObserver:删除观察者.

KVO键值观察

这是一个典型的观察者模式,观察者在键值改变时会得到通知.
NSKeyValueObserving非正式协议定义了一种机制,它允许对象去监听其它对象的某个属性的修改.NSObject提供了一个NSKeyValueObserving协议的默认实现,它为所有对象提供了一种自动发送修改通知的能力.我们可以通过禁用自动发送通知并使用这个协议提供的方法来手动实现通知的发送,以便更精确地去处理通知.

KVO的使用步骤

  • 注册需要观察的对象的属性addObserver:forKeyPath:options:context:.
  • 实现observeValueForKeyPath:ofObject:change:context:方法,这个方法当观察的属性变化时会自动调用.在这个方法中还通过NSKeyValueObservingOptionNew这个参数要求把新值在dictionary中传递过来.
  • 取消注册观察removeObserver:forKeyPath:context:.

手动设置KVO


//关闭某key值的KVO+ (BOOL)automaticallyNotifiesObserversForKey:(NSString *)key {BOOL automatic = YES;
if ([key isEqualToString:@"age"]) {automatic = NO;
}

else {automatic = [super automaticallyNotifiesObserversForKey:key];
}

return automatic;
}


//重写setter方法手动设置KVO- (void)setAge:(int)age {
//手动设置KVOif (_age != age) {[self willChangeValueForKey:@"age"];
_age = age;
[self didChangeValueForKey:@"age"];
}

}

KVO的实现原理

当某一个类的实例第一次使用KVO的时候,系统就会在运行期间动态的创建该类的一个派生类,该类的命名规则一般是以NSKVONotifying为前缀,以原本的类名为后缀。并且将原型的对象的isa指针指向该派生类。同时在派生类中重载了使用KVO的属性的setter方法,在重载的setter方法中实现真正的通知机制,正如前面我们手动实现KVO一样。这么做是基于设置属性会调用 setter 方法,而通过重写就获得了 KVO 需要的通知机制。当然前提是要通过遵循 KVO 的属性设置方式来变更属性值,如果仅是直接修改属性对应的成员变量,是无法实现 KVO 的。

猜你喜欢
理解netty对protocol buffers的编码解码 Android系统移植与驱动开发——第六章——使用实例来理解Linux驱动开发及心得
OC泛型 node.js 小爬虫 imooc 2016.03.06
Linux配置环境报“/usr/local/develop-tools/apache-maven-3.3.9/bin: 是一个目录“的解决方案 $(document).ready()即$()方法和window.onload方法的比较
[Effective JavaScript 笔记]第15条:当心局部块函数声明笨拙的作用域 html_css_js相关题.docx
Source Tree - ssl certificate problem unable to get local issuer certificate [Effective JavaScript 笔记] 第14条:当心命名函数表达式笨拙的作用域