美文网首页
iOS开发-消息传递方式-NotificationCenter

iOS开发-消息传递方式-NotificationCenter

作者: 张囧瑞 | 来源:发表于2018-03-08 21:34 被阅读92次

    说完了target-action

    说完了KVO

    今天说另一种消息传递的方式,这种方式和KVO很像,通常也是用于一对多的情况,这种消息传递的方式就是NotificationCenter

    NotificationCenter 翻译过来就是通知中心,他和我们生活中的广播很相似。

    如何使用NotificationCenter

    使用NotificationCenter和KVO非常相似,一般也是分为4个步骤:

    • 1.添加观察者。

    • 2.实现响应方法。

    • 3.发出通知。

    • 4.移除观察者。

    一般在使用NotificationCenter时我们通常都是直接使用系统默认的NotificationCenter

    
    @property (class, readonly, strong) NSNotificationCenter *defaultCenter;
    
    

    使用类方法调用他之后,我们就可以做响应的添加、发出通知、移除等操作了。

    添加观察者

    添加观察者有两种方式,一个是直接调用函数添加,然后再编写实现的方法,通过selector来响应通知。

    还有一种方法是使用block,在代码块内部实现响应通知的操作。

    直接添加

    直接添加观察者时可以通过函数:

    
    - (void)addObserver:(id)observer selector:(SEL)aSelector name:(nullable NSNotificationName)aName object:(nullable id)anObject;
    
    

    传递了响应的参数之后就完成了观察者的添加,那我们来看看传递的参数都有什么作用。

    observer

    通知的观察者,当通知post出来之后,观察者会收到响应的通知,不可以为nil。

    aSelector

    收到通知之后需要做的操作,当通知post出来之后,会触发该方法,当中会传递一个NSNotification的参数。

    aName

    通知名称,用来区分发送通知的对象,可以为nil,但是因为应用的执行的过程中会发出很多通知,如果为nil则代表接受所有的通知。

    anObject

    发送的对象,也是用来区分发送通知的对象,可以为nil,当为nil时,也会接受同一通知名称的所有的通知。

    一般情况下,如果我们将通知名称和发送对象都置为nil时,我们可以监听到当前通知中心的全部通知。比如:

    
    - (void)addNotificationCenter
    
    {
    
     [[NSNotificationCenter defaultCenter] addObserver:self selector:@selector(notificationCall:) name:nil object:nil];
    
     [[NSNotificationCenter defaultCenter] postNotificationName:@"postNotification" object:nil];
    
    }
    
    - (void)notificationCall:(NSNotification *)notification
    
    {
    
     NSLog(@"notification = %@", notification.name);
    
    }
    
    

    上边一段代码执行了之后我们会发现结果如下:

    
    2018-03-08 17:28:25.368264+0800 MessagePassingDemo[50635:3561130] notification = postNotification
    
    2018-03-08 17:28:25.368593+0800 MessagePassingDemo[50635:3561130] notification = UINavigationControllerWillShowViewControllerNotification
    
    2018-03-08 17:28:25.370494+0800 MessagePassingDemo[50635:3561130] notification = UIViewAnimationDidCommitNotification
    
    2018-03-08 17:28:25.370677+0800 MessagePassingDemo[50635:3561130] notification = UIViewAnimationDidStopNotification
    
    2018-03-08 17:28:25.371431+0800 MessagePassingDemo[50635:3561130] notification = _UIWindowContentWillRotateNotification
    
    2018-03-08 17:28:25.372607+0800 MessagePassingDemo[50635:3561130] notification = UIViewAnimationDidCommitNotification
    
    2018-03-08 17:28:25.372898+0800 MessagePassingDemo[50635:3561130] notification = UIViewAnimationDidStopNotification
    
    2018-03-08 17:28:25.373097+0800 MessagePassingDemo[50635:3561130] notification = _UIWindowContentWillRotateNotification
    
    2018-03-08 17:28:25.392586+0800 MessagePassingDemo[50635:3561130] notification = UIViewAnimationDidCommitNotification
    
    2018-03-08 17:28:25.393403+0800 MessagePassingDemo[50635:3561130] notification = UIViewAnimationDidCommitNotification
    
    2018-03-08 17:28:25.423827+0800 MessagePassingDemo[50635:3561130] notification = UIViewAnimationDidCommitNotification
    
    2018-03-08 17:28:25.424067+0800 MessagePassingDemo[50635:3561130] notification = UIViewAnimationDidStopNotification
    
    2018-03-08 17:28:25.424322+0800 MessagePassingDemo[50635:3561130] notification = UIViewAnimationDidCommitNotification
    
    2018-03-08 17:28:25.424721+0800 MessagePassingDemo[50635:3561130] notification = UIViewAnimationDidStopNotification
    
    2018-03-08 17:28:25.426606+0800 MessagePassingDemo[50635:3561130] notification = UIViewAnimationDidCommitNotification
    
    2018-03-08 17:28:25.426780+0800 MessagePassingDemo[50635:3561130] notification = UIViewAnimationDidStopNotification
    
    2018-03-08 17:28:25.427223+0800 MessagePassingDemo[50635:3561130] notification = UIViewAnimationDidCommitNotification
    
    2018-03-08 17:28:25.427401+0800 MessagePassingDemo[50635:3561130] notification = UIViewAnimationDidStopNotification
    
    2018-03-08 17:28:25.431615+0800 MessagePassingDemo[50635:3561130] notification = UIViewAnimationDidCommitNotification
    
    2018-03-08 17:28:25.790567+0800 MessagePassingDemo[50635:3561130] notification = UIViewAnimationDidCommitNotification
    
    2018-03-08 17:28:25.790769+0800 MessagePassingDemo[50635:3561130] notification = UIViewAnimationDidStopNotification
    
    2018-03-08 17:28:25.791427+0800 MessagePassingDemo[50635:3561130] notification = UIViewAnimationDidStopNotification
    
    2018-03-08 17:28:25.792659+0800 MessagePassingDemo[50635:3561130] notification = UIViewAnimationDidCommitNotification
    
    2018-03-08 17:28:25.793148+0800 MessagePassingDemo[50635:3561130] notification = UIViewAnimationDidStopNotification
    
    2018-03-08 17:28:25.935701+0800 MessagePassingDemo[50635:3561130] notification = UIViewAnimationDidStopNotification
    
    2018-03-08 17:28:25.936635+0800 MessagePassingDemo[50635:3561130] notification = UINavigationControllerDidShowViewControllerNotification
    
    2018-03-08 17:28:25.936938+0800 MessagePassingDemo[50635:3561130] notification = UIViewAnimationDidStopNotification
    
    

    所以一般情况下,添加观察者时,我们至少都会指定好通知的名称。

    
    - (void)addNotificationCenter
    
    {
    
     [[NSNotificationCenter defaultCenter] addObserver:self selector:@selector(notificationCall:) name:@"postNotification" object:nil];
    
     [[NSNotificationCenter defaultCenter] postNotificationName:@"postNotification" object:nil];
    
    }
    
    - (void)notificationCall:(NSNotification *)notification
    
    {
    
     NSLog(@"notification = %@", notification.name);
    
    }
    
    

    输出:

    
    2018-03-08 17:30:22.767448+0800 MessagePassingDemo[50709:3564576] notification = postNotification
    
    

    另外,如果我们设置了通知名,没有设置发送对象时,所有此通知名的通知都会被接收到,如果设置了发送对象:

    
    - (void)addNotificationCenter
    
    {
    
     [[NSNotificationCenter defaultCenter] addObserver:self selector:@selector(notificationCall:) name:@"postNotification" object:self.buttonA];    
    
    }
    
    - (void)buttonAClicked
    
    {
    
     [[NSNotificationCenter defaultCenter] postNotificationName:@"postNotification" object:self.buttonA];
    
    }
    
    - (void)buttonBClicked
    
    {
    
     [[NSNotificationCenter defaultCenter] postNotificationName:@"postNotification" object:self.buttonB];
    
    }
    
    

    这时候就只有buttonAClickedbuttonA被点击时才会触发通知。

    使用 block

    使用block添加观察者时可以调用如下函数:

    
    - (id <NSObject>)addObserverForName:(nullable NSNotificationName)name object:(nullable id)obj queue:(nullable NSOperationQueue *)queue usingBlock:(void (^)(NSNotification *note))block API_AVAILABLE(macos(10.6), ios(4.0), watchos(2.0), tvos(9.0));
    
    

    相比较于上一个方法,这种方法中没有指定具体的观察者,并且用block代替了触发的方法,而且新增了一个queue

    因为用block回调来替代了触发的方法,所以在发送通知时,也不需要去某个具体观察者的方法列表中去找对应的列表了,直接执行block中的内容就可以了。

    比如上边的方法,我们就可以改成这样:

    
     [[NSNotificationCenter defaultCenter] addObserverForName:@"postNotification" object:nil queue:nil usingBlock:^(NSNotification * _Nonnull note) {
    
     NSLog(@"notification = %@", note.name);
    
     }];
    
     [[NSNotificationCenter defaultCenter] postNotificationName:@"postNotification" object:nil];
    
    

    这样写,可以让代码看起来更加紧凑。

    而新增的queue,在默认情况下在post的线程中处理,比如这段代码:

    
    - (void)addNotificationCenter
    
    {    
    
     [[NSNotificationCenter defaultCenter] addObserverForName:@"postNotification" object:nil queue:nil usingBlock:^(NSNotification * _Nonnull note) {
    
     NSLog(@"notification = %@", note.name);
    
     NSLog(@"receive thread = %@", [NSThread currentThread]);
    
     }];
    
     [[NSNotificationCenter defaultCenter] postNotificationName:@"postNotification" object:nil];
    
     dispatch_async(dispatch_get_global_queue(DISPATCH_QUEUE_PRIORITY_DEFAULT, 0), ^{
    
     NSLog(@"post thread = %@", [NSThread currentThread]);
    
     [[NSNotificationCenter defaultCenter] postNotificationName:@"postNotification" object:nil];
    
     });
    
    }
    
    

    打印出来我们就会发现:

    
    2018-03-08 17:49:54.732746+0800 MessagePassingDemo[51217:3595209] notification = postNotification
    
    2018-03-08 17:49:54.733093+0800 MessagePassingDemo[51217:3595209] receive thread = <NSThread: 0x60000006e780>{number = 1, name = main}
    
    2018-03-08 17:49:54.733711+0800 MessagePassingDemo[51217:3595267] post thread = <NSThread: 0x6000004639c0>{number = 3, name = (null)}
    
    2018-03-08 17:49:54.734052+0800 MessagePassingDemo[51217:3595267] notification = postNotification
    
    2018-03-08 17:49:54.734320+0800 MessagePassingDemo[51217:3595267] receive thread = <NSThread: 0x6000004639c0>{number = 3, name = (null)}
    
    

    除了这部分的不同之外,剩下的两者应该没有什么不同的地方了,还有注意一点就是第二种方法中的循环引用问题的发生。

    实现响应方法

    响应方法这边就不多说了,一种是通过调用,一种是通过block,但是两种都会传递一个NSNotification对象,注意不是NSNotificationCenter,是NSNotification

    其中包括了

    
    @property (readonly, copy) NSNotificationName name;
    
    @property (nullable, readonly, retain) id object;
    
    @property (nullable, readonly, copy) NSDictionary *userInfo;
    
    

    一般情况下,我们可以通过userInfo来传递一些信息。

    post 通知

    post通知一共有三种方法。

    
    - (void)postNotification:(NSNotification *)notification;
    
    - (void)postNotificationName:(NSNotificationName)aName object:(nullable id)anObject;
    
    - (void)postNotificationName:(NSNotificationName)aName object:(nullable id)anObject userInfo:(nullable NSDictionary *)aUserInfo;
    
    

    同理,name和object也可以为nil,只不过发出去的消息并没有理你罢了。另外userInfo用来传递一些简单的信息。

    比如:

    
    - (void)addNotificationCenter
    
    {
    
     [[NSNotificationCenter defaultCenter] addObserver:self selector:@selector(notificationCall:) name:@"postNotification" object:nil];
    
     [[NSNotificationCenter defaultCenter] postNotificationName:@"postNotification" object:nil userInfo:@{@"name":@"zhangsan",@"age":@"18"}];
    
    }
    
    
    
    - (void)notificationCall:(NSNotification *)notification
    
    {
    
     NSLog(@"notification = %@", notification.name);
    
     NSLog(@"userinfo = %@",notification.userInfo);
    
    }
    
    

    这边就会输出:

    
    2018-03-08 17:58:14.998128+0800 MessagePassingDemo[51452:3608470] notification = postNotification
    
    2018-03-08 17:58:14.998407+0800 MessagePassingDemo[51452:3608470] userinfo = {
    
     age = 18;
    
     name = zhangsan;
    
    }
    
    

    移除观察者

    移除观察者一共有两种方法:

    
    - (void)removeObserver:(id)observer;
    
    - (void)removeObserver:(id)observer name:(nullable NSNotificationName)aName object:(nullable id)anObject;
    
    

    第一个方法是不针对某个特定通知,移除全部观察者,第二种是有针对性的移除某个观察者。而如果在第二种方法中name和object参数都传递nil,就和第一种方法完全一样了。

    一般在dealloc中会使用第一种方法移除全部的观察者。而在viewWillDisappear中则使用第二种方式移除。

    这里注意,如果我们不在合适的时机移除观察者,导致添加了重复的观察者的话,新注册的已有名字的观察者并不会覆盖之前的观察者,而是会添加两个观察者,这会导致post时响应两次或更多,比如:

    
    - (void)addNotificationCenter
    
    {
    
     [[NSNotificationCenter defaultCenter] addObserver:self selector:@selector(notificationCall:) name:@"postNotification" object:nil];
    
     [[NSNotificationCenter defaultCenter] addObserver:self selector:@selector(notificationCall:) name:@"postNotification" object:nil];
    
     [[NSNotificationCenter defaultCenter] postNotificationName:@"postNotification" object:nil userInfo:@{@"name":@"zhangsan",@"age":@"18"}];
    
    }
    
    
    
    - (void)removeNotificationCenter
    
    {
    
     [[NSNotificationCenter defaultCenter] removeObserver:self];
    
     [[NSNotificationCenter defaultCenter] removeObserver:self name:nil object:nil];
    
    }
    
    

    打印的结果:

    
    2018-03-08 21:27:30.566903+0800 MessagePassingDemo[52218:3651551] notification = postNotification
    
    2018-03-08 21:27:30.567124+0800 MessagePassingDemo[52218:3651551] userinfo = {
    
     age = 18;
    
     name = zhangsan;
    
    }
    
    2018-03-08 21:27:30.567253+0800 MessagePassingDemo[52218:3651551] notification = postNotification
    
    2018-03-08 21:27:30.567440+0800 MessagePassingDemo[52218:3651551] userinfo = {
    
     age = 18;
    
     name = zhangsan;
    
    }
    
    

    最后

    NotificationCenter的简单使用就这么多,以上这些内容仅供个人学习使用,如果有什么不对的地方还请各位大佬多多指教。

    相关文章

      网友评论

          本文标题:iOS开发-消息传递方式-NotificationCenter

          本文链接:https://www.haomeiwen.com/subject/yukufftx.html