iOS开发--事件监听处理
这是我个人的学习笔记 , 如有不同见解欢迎评论交流 .
( GeekBand-极客班 http://geekband.com )
( 我的微博 : http://weibo.com/JohnnyB0Y )
iOS开发中, 事件监听有很多种方法 . 下面是我个人的见解 .
代理模式
-
什么是代理
1.设计拆分,方便重用.
2.清晰定义功能,变化行为/自定义界面.
3.松散耦合,容易扩展.
-
什么时候用代理
1.开发中,原对象捕捉到了事件但无法处理,就需要通知有能力处理这件事的代理对象来处理.
2.代理通常是一对一的(一份协议搞很多代理是很少见的,可以用NSNotification)
3.原对象和代理对象是解耦的,意味着有能力处理协议任务就能做我的代理.( 关键词: 解耦 )
-
怎么用代理
1.定个协议先. (申明代理原型)
2.联络代理的方式. (申明代理变量)
3.确定什么时候通知代理. (调用代理方法)
4.然后让代理遵守协议.
5.拿到代理的联络方式. (设置代理的值)
6.让代理实现代理方法 .
-
代理的补充
1.由于解耦了,原对象和代理对象之间的数据传递就比较依赖于代理方法.
2.通常原对象会把自己当方法参数传进去 , 而代理对象通过方法返回值来向原对象反馈信息.(UITableView的数据源方法就是需要代理反馈的信息和数据)
3.当代理挂掉的时候要移除代理,当然如果代理属性使用了weak修饰词就不需要这么做了.
KVO 键值观察
-
什么是 KVO
1.KVO有框架级别的支持,不需要你去写太多代码就能观察属性的改变获得通知.
2.我比较喜欢把KVO比作检测器,例如温度计 (周围的温度变化 , 温度计会做出变化)
3.KVO支持一对一,一对多,多对一. ( 多人监听一个属性,一个人监听多个属性.....)
-
什么时候用 KVO
1.controller对象观察model对象的属性改变,view对象通过controller观察model对象的属性改变.
2.观察属性改变也可以用代理,可是麻烦多多.所以用KVO是最简便快捷的.
-
怎么用 KVO
1.确定哪个对象要观察哪个对象的属性改变 [被观察者 addObserver:观察者 forKeyPath:属性的路径 options:属性改变前后值选择 context:想传递的对象]; 2.观察者重写键值观察方法.( 当属性改变,自动调用这个方法 ) - (void)observeValueForKeyPath:(NSString *)keyPath ofObject:(id)object change:(NSDictionary *)change context:(void *)context;
-
KVO的补充
1.有监听,就有移除监听.
[被观察者 removeObserver:观察者 forKeyPath:属性路径];
[被观察者 removeObserver:观察者 forKeyPath:属性路径 context:想传递的对象];
NSNotification 通知
-
什么是 NSNotification
1.NSNotification相当于广播,谁想知道发生什么事情就去接收广播.
2.通知机制的核心是通知中心,它是个系统级别的东西.
3.所有的通知都由通知中心发出.
4.接受通知也需要在通知中心接收.
-
什么时候用 NSNotification
1.通知可以一对一,一对多,它跟接收通知的对象可以完全没有联系,比代理还要解耦.
2.通常一对多就使用通知.(当然通知在性能上比代理和KVO差很多)
-
怎么用 NSNotification
1.发送通知
1.1当前线程发送通知
- (void)postNotificationName:通知的名称 object:发送通知的对象 userInfo:传送数据的字典;
1.2异步线程发送通知
NSNotification *note = [NSNotification notificationWithName:通知名 object:nil userInfo:nil];
[[NSNotificationQueue defaultQueue] enqueueNotification:note
postingStyle:发送时机 (枚举值 一般选NSPostNow)
coalesceMask:合并标记 (枚举值 一般选NSNotificationNoCoalescing)
forModes:运行循环数组 (传nil代表NSDefaultRunLoopMode)];
2.接收通知
2.1当前线程调用方法子
- addObserver:(id)notificationObserver // 接收通知的观察者
selector:(SEL)notificationSelector // 接收通知后调用的方法
name:(NSString *)notificationName // 通知的名称
object:(id)notificationSender; // 发送通知的对象
2.2异步线程调用Block
- addObserverForName:(NSString *)name // 通知的名称
object:(id)obj // 发送通知的对象
queue:(NSOperationQueue *)queue // 处理Block的线程队列
usingBlock:(void (^)(NSNotification *note))block;// 要执行的Block (也是接收通知的观察者)
3.移除通知
NSNotification *note = [NSNotification notificationWithName:通知名 object:nil userInfo:nil];
[[NSNotificationQueue defaultQueue] dequeueNotificationsMatching:note coalesceMask:合并标记];
[[NSNotificationCenter defaultCenter] removeObserver:接收通知者];
[[NSNotificationCenter defaultCenter] removeObserver:接收通知者 name:通知名 object:通知的发出者];
-
通知的补充
1.接收通知的对象被销毁前一定要移除通知.
2.利用发通知和接收通知的线程能力,可以有效防止响应不及时等问题.
3.由于通知资源开销比较大,通知发送者和接受者没有必然联系debug非常困难,所以要谨慎使用.
Block 代码块
-
什么是 Block
-
什么时候用 Block
-
怎么用 Block
总结
- 苹果说这些都是iOS开发的大绝. 我们一定要好好利用.