【前言介绍】

iOS的一种设计模式,观察者Observer模式(也叫发布/订阅,即Publich/Subscribe模式)。

观察者模式,包含了通知机制(notification)和KVO(Key-value-observing)机制。

在这本文中,我们将介绍在日常项目当中经常使用到的通知机制这一种设计模式。

通知机制

委托机制是代理“一对一”的对象之间的通信,而通知机制是广播“一对多”的对象之间的通信;

一、是什么?【生活问题例子】

“短信天气预报”

当A类发送一条信息给通知中心时,注册为用户(观察者)的B类群就会收到相应的通知,并作出反应。

二、有什么用?【代码中的应用】

在不同类之间如何传递数据?

有几种方法:属性传递、代理协议,另外就是通知。

通知:在A类中创建的方法,B类中执行,且可以使用该通知携带数据传递给对方;

三、有什么不同?【与其他“通知”的不同?】

经常提到的通知,有“广播通知”、“本地通知”、“推送通知”

本文所介绍的就是广播通知,是实现观察者模式的一种机制,可以在一个应用中的多个对象之间进行通信传递数据。

而本地通知和推送通知主要是给用户发送“通知提示”,例如警告提示、声音、震动以及如图标上的红色数字提示。

第一种由“本地发送通知”给用户,第二种由第三方应用发送给苹果官方的远程服务器,然后再由服务器“推送通知”给用户。

四、产品经理:老规矩,代码拿来~【具体实现】

过程:

  • 在通知机制中,需要(或者说感兴趣)接收某个通知的信息的所有对象都可以成为接收者,首先注册成为观察者。
  • 进行注册后,通知中心就会把发布者发送的通知信息,广播给注册过该通知的观察者。且观察者只能接收到通知中心的信息,不能知道通知是谁投送的。
  • 最后,接受者不想再对关注该通知的信息时,可以给通知中心发生解除注册的信息,之后都不再接收到通知了。

1.获取通知中心(NSNotificationCenter)对象:(就像获取移动营运商短信中心的权限,作为媒介)

发布、注册、解除通知都需要使用通知中心,负责协助不同对象、不同类之间的消息通信。

[NSNotificationCenter  defaultCenter];   //需要注意的是,通知中心也是一个单例

2.发布(A类)和接收(B类)

a.做为发布者的A类发送通知:

可以使用一下三个方法:

- (void)postNotification:(NSNotification *)notification;

- (void)postNotificationName:(NSString *)aName object:(id)anObject;

- (void)postNotificationName:(NSString *)aName object:(id)anObject userInfo:(NSDictionary *)aUserInfo;
  • postNotificationName:指定消息名称;
  • object:指定发消息者;
  • userInfo:通知中用于传递参数的载体,传递的方法是把参数放在NSDictionary类型的userInfo中。例如:NSDictionary *dict = [notification userInfo];

一般使用第三个方法,如果参数不需要的,可以设置为nil.

b.注册通知,加入观察者:

做为观察者B类注册通知,进行监听:

- (void)addObserver:(id)observer selector:(SEL)aSelector name:(nullable NSString *)aName object:(nullable id)anObject;
 //@selector中为回调方法,在本类中对通知进行相应的处理,name为通知名称、object为对象;

剖析:

  • object == nil,那么客户对象(self)将收到任何对象发出NSWindowDidBecomeMainNotification的通知消息;
  • name == nil,那么观察者将接收到object对象的所有消息,但是确定不了接收这些消息的顺序。
  • object == nil,name == nil,那么该观察者将收到所有对象的所有消息。

对于一个任意的观察者observer,如果不能保证其对应的selector有本类自定义的方法(例如:MyMethod),可采用[observer respondsToSelector:@selector(MyMethod:)]] 进行检查。

所以完整的添加观察者过程为:

if([observer respondsToSelector:@selector(MyMethod:)]) {
[[NSNotificationCenter defaultCenter] addObserver:observer selector:@selector(MyMethod:) name:NSWindowDidBecomeMainNotification object:nil];
}

当然在苹果API中也有另外一个注册观察者的方法:

- (id <NSObject>)addObserverForName:(nullable NSString *)name object:(nullable id)obj queue:(nullable NSOperationQueue *)queue usingBlock:(void (^)(NSNotification *note))block

此方法是支持在该方法中进行block回调的,而queue参数就是表示此模块在queue队列中进行。

但是这方法一般不采用,因为在此方法的 block 中,稍微不注意调用 self 的话,会引起循环引用,造成内存泄露,所以还是建议使用第一种方法进行观察者的创建。

c.移除通知

注册过的对象必须在释放之前注销掉,如果不这样的话,当该通知再次出现时,通知中心可能会向已释放的观察者对象发送消息,从而导致应用崩溃。

在ARC下,系统会自动回收无用的通知对象内存,但是由于系统回收机制ARC有一定的延迟性,所以即使不会出错,也建议养成习惯,对通知进行手动释放无用的通知。
移除有2种方法:

//释放所有通知
- (void)removeObserver:(id)observer;
//释放名称为aName的通知
- (void)removeObserver:(id)observer name:(nullable NSString *)aName object:(nullable id)anObject;

一般在视图控制器中,可以在“didReceiveMemoryWarning:”中发送解除消息:【这只是参考,建议还是在 :-(void)dealloc ){} 中进行移除。 】

-(void)didReceiveMemoryWarning
{
[super didReceiveMemoryWarning];
//移除观察者
[[NSNotificationCenter defaultCenter]removeObserver:self];
}

(by:从 iOS 9 开始通知中心会对观察者进行 weak 弱引用,所以不需要在观察者对象释放之前从通知中心移除。即使不对通知进行手动移除,指针也会在注册者被回收后自动置空,向空指针 nil 发送消息是不会有问题的。
但是,通过 - (id )addObserverForName: object: queue: usingBlock: 方法注册的观察者依然需要手动的释放,因为通知中心对它们持有的是强引用。)

五、那些年我们用过的系统通知名称~

系统自带的也有许多有用的通知,我们只需要注册为相应的通知接收对象,就能根据通知状态的变化发生相应的数据改变。

部分系统通知名称如下:

UIApplicationDidFinishLaunchingNotification // 应用程序启动后
UIApplicationDidBecomActiveNotification // 进入前台
UIApplicationWillResignActiveNotification // 应用将要进入后台
UIApplicationDidEnterBackgroundNotification // 进入后台
UIKeyboardWillShowNotification // 键盘即将显示
UIKeyboardDidShowNotification // 键盘显示完毕
UIKeyboardWillHideNotification // 键盘即将隐藏
UIKeyboardDidHideNotification // 键盘隐藏完毕

六、举个栗子:“

iOS 趣谈设计模式——通知的更多相关文章

  1. IOS开发常用设计模式

    IOS开发常用设计模式 说起设计模式,感觉自己把握不了笔头,所以单拿出iOS开发中的几种常用设计模式谈一下. 单例模式(Singleton) 概念:整个应用或系统只能有该类的一个实例 在iOS开发我们 ...

  2. [转]ASP.NET应用程序生命周期趣谈(五) IIS7瞎说

    Ps:建议初学者在阅读本文之前,先简要了解一下之前的几篇文章,以便于熟悉本文提到的一些关于IIS6的内容,方便理解.仅供参考. PS:为什么叫瞎说呢?我觉得自己理解的并不到位,只能是作为一个传声筒,希 ...

  3. iOS常用的设计模式

    iOS常用的设计模式有:单例模式.委托模式.观察者模式和MVC模式.下面分别简单介绍. 一:单例模式 我们常用的UIApplication.NSUserdefaults.NSNotificationC ...

  4. [转]ASP.NET应用程序生命周期趣谈(三) HttpModule

    在之前的文章中,我们提到过P_Module(HttpModule)这个能干的程序员哥们儿,它通过在项目经理HttpApplication那里得到的授权,插手整个应用程序级别的事件处理.所有的HttpM ...

  5. 【iOS 单例设计模式】底层解析与运用

    [iOS 单例设计模式]底层解析与运用 一.单例设计名词解释: (官方解释)单例模式确保一个类只有一个实例,自行提供这个实例并向整个系统提供这个实例.(形象比喻)程序 — 公司   单例实例 - 管理 ...

  6. 趣谈、浅析CRLF和LF

    作为程序员,在处理文件和输入输出的时候经常要跟CRLF和LF打交道.可能大家多少知道一些,但总是记不清楚,我也是这样的,因此写下这篇博文,作为记录. 首先,明确他们的意思:CR(回车),LF(换行). ...

  7. iOS 10推送通知开发

    原文地址:Developing Push Notifications for iOS 10,译者:李剑飞 虽然通知经常被过度使用,但是通知确实是一种获得用户关注和通知他们需要更新或行动的有效方式.iO ...

  8. (七十三)iOS本地推送通知的实现

    iOS的推送通知分为本地推送和网络推送两种,如果App处于挂起状态,是可以发送本地通知的,如果已经被杀掉,则只有定时通知可以被执行,而类似于QQ的那种网络消息推送就无法实现了,因为App的网络模块在被 ...

  9. 趣谈生成函数 =v=

    趣谈生成函数 =v= 今天luyouqi在洛谷随机跳题rand出来一道生成函数板子题,然后我给做了(雾 发现小伙伴们还不会生成函数,于是我试着写这篇生成函数简介.(其实我也不怎么会生成函数这么高级的东 ...

随机推荐

  1. Unix系统小知识(转)

    Unix操作系统的小知识 2.VI添加行号/翻页/清屏 .在对话模式时(即输完Esc再输入: ),输入“:set number”可以将编辑的文本加上行号.跟玩俄罗斯方块一样方便的上下左右移动箭头的快捷 ...

  2. 补交作业-第八周PSP

    一.表格 C(分类) C(内容) S(开始时间) ST(结束时间) I(打断时间) △(净工作时间) 讨论 用户界面 9:30 10:40 15 55 编码 编码 13:20 16:30 10 180 ...

  3. Android 中 非对称(RSA)加密和对称(AES)加密

    在非对称加密中使用的主要算法有:RSA.Elgamal.背包算法.Rabin.D-H.ECC(椭圆曲线加密算法)等. 优点: 非对称加密与对称加密相比,其安全性更好:对称加密的通信双方使用相同的秘钥, ...

  4. c++宏使用总结【转】

    C/C++中宏总结C程序的源代码中可包括各种编译指令,这些指令称为预处理命令.虽然它们实际上不是C语言的一部分,但却扩展了C程序设计的环境. ANSI标准定义的C语言预处理程序包括下列命令:  #de ...

  5. Android应用内存泄漏的定位、分析与解决策略

    什么是内存泄漏 对于不同的语言平台来说,进行标记回收内存的算法是不一样的,像 Android(Java)则采用 GC-Root 的标记回收算法.下面这张图就展示了 Android 内存的回收管理策略( ...

  6. C# Marshal.GetActiveObject() 遭遇 HRESULT:0x800401E3 (MK_E_UNAVAILABLE))

    解决办法: 勿以管理员权限运行Visual Studio

  7. JS 深度拷贝 Object Array

    JS 深度拷贝 Object Array function cloneObj(o) { var isArray = o instanceof Array; var isObject = o insta ...

  8. 【转】JSch - Java实现的SFTP(文件上传详解篇)

    JSch是Java Secure Channel的缩写.JSch是一个SSH2的纯Java实现.它允许你连接到一个SSH服务器,并且可以使用端口转发,X11转发,文件传输等,当然你也可以集成它的功能到 ...

  9. 安装中文版cacti监控华为交换机流量并实现95计费

    摘要:一. 装置yum源: 以网易yum源为例 1. 下载repo文件 下载地点:http://mirrors.163.com/.help/CentOS6-Base-163.repo 2.备份并调换体 ...

  10. js转换数据类型为浮点型,并取两位小数点

    转换数据类型 parseFloat();//转换为浮点型 parseInt();//转换为整形 取后面两位小数 bianliang.toFixed(2);//取后面两位小数,2代表取多少位