美文网首页
通知中心

通知中心

作者: 前年的邂逅_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];
    }
    

    相关文章

      网友评论

          本文标题:通知中心

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