美文网首页
通知中心

通知中心

作者: 前年的邂逅_Jerry | 来源:发表于2017-03-03 15:39 被阅读179次

一、主线程开启通知中心

在主线程发送通知是同步的,执行顺序 before、ing、after。

- (void)viewDidLoad {
    [super viewDidLoad];
    //注册通知
    //object传nil,表示,接受所有名字为notifyName的通知,否则只接受指定object对象的通知。
    [[NSNotificationCenter defaultCenter] addObserver:self selector:@selector(handleNotifi:) name:@"notifyName" object:nil];
}
- (void)handleNotifi:(NSNotification*)notifi{
    NSLog(@"ing");
    NSLog(@"%s",__func__);
}
- (IBAction)btnClicked:(id)sender {
    NSLog(@"before");
    //发送消息
    [[NSNotificationCenter defaultCenter] postNotificationName:@"notifyName" object:nil];
    NSLog(@"after");
}

在dealloc中移除通知中心。注:在控制器中注册了一个消息中心,可以不用在dealloc中移除通知,因为控制器自动执行了这一操作。如果,自己创建一个类,继承于NSObject类,需要手动移除通知,否则会在iOS9以下的系统里崩溃。

//移除所有的通知
- (void)removeObserver:(id)observer;
//移除指定的通知
- (void)removeObserver:(id)observer name:(nullable NSNotificationName)aName object:(nullable id)anObject;

二、子线程中开启通知中心

执行的顺序是before、after、ing。

- (void)viewDidLoad {
    [super viewDidLoad];
    [[NSNotificationCenter defaultCenter] addObserver:self selector:@selector(handleNotifi:) name:@"notifyName" object:nil];   
}
- (void)handleNotifi:(NSNotification *)notify{
    NSLog(@"ing");
}
- (IBAction)btnClicked:(id)sender {
    NSLog(@"before");
    [NSThread detachNewThreadWithBlock:^{
        [[NSNotificationCenter defaultCenter] postNotificationName:@"notifyName" object:nil userInfo:@{@"name":@"ads"}];
    }];
    NSLog(@"after");
}

三、在主线程中用通知队列的形式开启通知

执行顺序:before、after、ing

- (void)viewDidLoad {
    [super viewDidLoad];
    [[NSNotificationCenter defaultCenter] addObserver:self selector:@selector(handleNotifi:) name:@"notifyName" object:nil];
}
- (void)handleNotifi:(NSNotification *)notify{
    NSLog(@"ing = %@",[NSThread currentThread]);
}
- (IBAction)btnClicked:(id)sender {
    [self notiQueue];
}
- (void)notiQueue{
    NSLog(@"before");   
    NSNotificationQueue * notiQueue = [NSNotificationQueue defaultQueue];
    NSNotification * noti = [NSNotification notificationWithName:@"notifyName" object:self userInfo:@{@"userInfo":@"name1"}];
    //将通知放入队列中,先进先出FIFO
    //NSPostWhenIdle 空闲时发送
    //NSPostASAP as soon as posible 尽快发送 当前runloop不是同一个mode,现等待一个runloop模式完后,就会执行通知事件
    //NSPostNow 马上执行
    [notiQueue enqueueNotification:noti postingStyle:NSPostWhenIdle coalesceMask:NSNotificationCoalescingOnName forModes:nil];
    NSLog(@"after");
}

四、在子线程中用通知队列开启通知

执行顺序:before、after
怎么不执行handleNotifi呢?原因:runloop没有资源。

- (void)viewDidLoad {
    [super viewDidLoad];
    [[NSNotificationCenter defaultCenter] addObserver:self selector:@selector(handleNotifi:) name:@"notifyName" object:nil];
}
- (void)handleNotifi:(NSNotification *)notify{
    NSLog(@"ing = %@",[NSThread currentThread]);
}
- (IBAction)btnClicked:(id)sender {
    [NSThread detachNewThreadWithBlock:^{
        [self notiQueue];
    }];
}
- (void)notiQueue{
    NSLog(@"before");   
    NSNotificationQueue * notiQueue = [NSNotificationQueue defaultQueue];
    NSNotification * noti = [NSNotification notificationWithName:@"notifyName" object:self userInfo:@{@"userInfo":@"name1"}];
    //将通知放入队列中,先进先出FIFO
    //NSPostWhenIdle 空闲时发送
    //NSPostASAP as soon as posible 尽快发送 当前runloop不是同一个mode,现等待一个runloop模式完后,就会执行通知事件
    //NSPostNow 马上执行
    [notiQueue enqueueNotification:noti postingStyle:NSPostWhenIdle coalesceMask:NSNotificationCoalescingOnName forModes:[NSArray arrayWithObjects:NSDefaultRunLoopMode,nil]];
    NSLog(@"after");
}

将runloop中添加资源,NSLog(@"runloop over"); 永远不会被执行。每次点击按钮,开启通知,每次点击都会多产生一条线程,这样子会产生资源浪费。
执行顺序:before、after、ing。

- (void)notiQueue{
    NSLog(@"before");
    NSNotificationQueue * notiQueue = [NSNotificationQueue defaultQueue];
    NSNotification * noti = [NSNotification notificationWithName:@"notifyName" object:self userInfo:@{@"userInfo":@"name1"}];
    //将通知放入队列中,先进先出FIFO
    //NSPostWhenIdle 空闲时发送
    //NSPostASAP as soon as posible 尽快发送 当前runloop不是同一个mode,现等待一个runloop模式完后,就会执行通知事件
    //NSPostNow 马上执行
    [notiQueue enqueueNotification:noti postingStyle:NSPostWhenIdle coalesceMask:NSNotificationCoalescingOnName forModes:[NSArray arrayWithObjects:NSDefaultRunLoopMode,nil]];
    NSLog(@"after");
    NSPort * port = [[NSPort alloc] init];
    [[NSRunLoop currentRunLoop] addPort:port forMode:NSRunLoopCommonModes];   
    [[NSRunLoop currentRunLoop] run];
    NSLog(@"runloop over");
}

五、在指定线程上处理

通过NSPort让通知在指定线程上执行呢?如果将NSPort 放在主线程,那么通知就会在主线程上执行,将NSPort加在子线程上通知事件就会在子线程。

- (void)viewDidLoad {
    [super viewDidLoad];
    myPort = [[NSPort alloc] init];
    [myPort setDelegate:self];
    //放到子线程上
    [NSThread detachNewThreadWithBlock:^{
        [[NSRunLoop currentRunLoop] addPort:myPort forMode:NSDefaultRunLoopMode];
        [[NSRunLoop currentRunLoop] run];
    }];
    //放在主线程上
    //[[NSRunLoop currentRunLoop] addPort:myPort forMode:NSDefaultRunLoopMode];
}
- (void)handlePortMessage:(NSPortMessage *)message{
    NSObject * obj = (NSObject *)message;
    NSLog(@"%@",[NSThread currentThread]);
    NSLog(@"message = %@",[obj valueForKey:@"msgid"]);
}
- (void)myPort{
    [myPort sendBeforeDate:[NSDate date] msgid:111 components:nil from:nil reserved:0];
}
- (IBAction)btnClicked:(id)sender {
    [self myPort];
}

相关文章

  • 通知中心

    ** 通知机制(消息机制)是一个应用程序级别的操作UIApplication 通知中心实际上是iOS程序内部之间的...

  • 通知中心

    通知中心 通知中心(NSNotificationCenter)每一个应用程序都有一个通知中心(NSNotifica...

  • 通知中心

    一、主线程开启通知中心 在主线程发送通知是同步的,执行顺序 before、ing、after。 在dealloc中...

  • 通知中心

    每个应用中只会有一个通知中心对象:NSNotificationCenter,通过单例方法,创建唯一对象,通知中心对...

  • NSNotification-通知

    通知 通知中心 NSNotificationCenter 通知中心的作用 添加观察者 通知的发送 给通知找到对应的...

  • 通知相关

    通知中心 通知中心(NSNotificationCenter)每一个应用程序都有一个通知中心(NSNotifica...

  • iOS 通知 与 通知中心

    iOS 通知中心:自己实现了一套消息机制,可以跨页面调用 类似Unity的SendMessage,是订阅、发布者模...

  • NotificationCenter 通知中心

    经测试发现1,对于默认的通知中心 NotificationCenter.default, 没有移除监听通知的情况...

  • 9.4、通知中心

    import UIKit class OneViewController: UIViewController {/...

  • IOS通知中心

    通知中心(NSnotificationCenter): 由发送者1或者多个发送者n 通过发送一个通知到通知中心,接...

网友评论

      本文标题:通知中心

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