iOS通知(Notification)使用淺談

iOS的通知(Notification)基本上都是以sync(同步)的方式發(fā)送的音诈,也就是說當(dāng)發(fā)送通知出去以后會先阻塞當(dāng)前線程,等到所有注冊該通知者接收到此通知并在當(dāng)前線程進行完相對應(yīng)的處理后花椭,程序才會繼續(xù)往下執(zhí)行。

為了測試這點忘朝,我們可以創(chuàng)建BYNotifyPoster和BYNotifyReceiver兩個類择同,分別負責(zé)發(fā)送和接收通知。在測試過程轨奄,我們分別在主線程和異步線程各發(fā)送一次通知孟害,測試接收者執(zhí)行的情況。

BYNotifyReceiver

//
//  BYNotifyReceiver.h
//  NotificationTest
//
//  Created by Beryter on 2017/12/23.
//  Copyright ? 2017年 Beryter. All rights reserved.
//

#import <Foundation/Foundation.h>

@interface BYNotifyReceiver : NSObject

@end


#import "BYNotifyReceiver.h"

@implementation BYNotifyReceiver

- (instancetype)init
{
    if (self = [super init]) {
        [[NSNotificationCenter defaultCenter] addObserver:self selector:@selector(handleNotification:) name:@"BYNotificationTest" object:nil];
    }
    return self;
}

/*!
 * @brief   接收通知(BYNotificationTest)
 * @param   note NSNotification通知對象
 */
- (void)handleNotification:(NSNotification *)note
{
    if ([[NSThread currentThread] isMainThread]) {
        NSLog(@"接收到通知 - BYNotificationTest挪拟,主線程挨务,開始處理相關(guān)邏輯");
    } else {
        NSLog(@"接收到通知 - BYNotificationTest,異步線程玉组,開始處理相關(guān)邏輯");
    }
}

- (void)dealloc
{
    [[NSNotificationCenter defaultCenter] removeObserver:self name:@"BYNotificationTest" object:nil];
}

@end

BYNotifyPoster

//
//  BYNotifyPoster.h
//  NotificationTest
//
//  Created by Beryter on 2017/12/23.
//  Copyright ? 2017年 Beryter. All rights reserved.
//

#import <Foundation/Foundation.h>

@interface BYNotifyPoster : NSObject

- (void)postNotification;

@end

#import "BYNotifyPoster.h"

@implementation BYNotifyPoster

/*!
 * @brief 發(fā)送通知BYNotificationTest
 */
- (void)postNotification
{
    NSLog(@"準備發(fā)送通知 - BYNotificationTest");
    [[NSNotificationCenter defaultCenter] postNotificationName:@"BYNotificationTest" object:nil];
    NSLog(@"BYNotificationTest - 通知發(fā)送完畢谎柄,繼續(xù)往下執(zhí)行");
    // TODO:發(fā)送通知完畢后,繼續(xù)執(zhí)行
}

@end

我們在測試ViewController中加入以下代碼:

- (void)viewDidAppear:(BOOL)animated
{
    [super viewDidAppear:animated];
    
    NSLog(@"------------主線程發(fā)送通知---------------");
    [self.poster postNotification];
    
    sleep(5);//延遲五秒
    
    NSLog(@"------------異步線程發(fā)送通知--------------");
     __weak typeof(self) weakSelf = self;
    dispatch_queue_t queue = dispatch_queue_create("NOTIFICATION_TEST_QUEUE", NULL);
    dispatch_async(queue, ^{
        [weakSelf.poster postNotification];
    });
}

運行后惯雳,測試結(jié)果如下:

2017-12-23 14:25:37.951231+0800 NotificationTest[4327:320244] ------------主線程發(fā)送通知---------------
2017-12-23 14:25:37.951388+0800 NotificationTest[4327:320244] 準備發(fā)送通知 - BYNotificationTest
2017-12-23 14:25:37.951496+0800 NotificationTest[4327:320244] 接收到通知 - BYNotificationTest朝巫,主線程,開始處理相關(guān)邏輯
2017-12-23 14:25:37.951604+0800 NotificationTest[4327:320244] BYNotificationTest - 通知發(fā)送完畢石景,繼續(xù)往下執(zhí)行
2017-12-23 14:25:42.952721+0800 NotificationTest[4327:320244] ------------異步線程發(fā)送通知--------------
2017-12-23 14:25:42.953099+0800 NotificationTest[4327:320290] 準備發(fā)送通知 - BYNotificationTest
2017-12-23 14:25:42.953337+0800 NotificationTest[4327:320290] 接收到通知 - BYNotificationTest劈猿,異步線程拙吉,開始處理相關(guān)邏輯
2017-12-23 14:25:42.953516+0800 NotificationTest[4327:320290] BYNotificationTest - 通知發(fā)送完畢,繼續(xù)往下執(zhí)行

根據(jù)運行后打印的log日志我們可以看到糙臼,BYNotifyPoster在發(fā)送完通知后庐镐,會等待所有被通知到的對象處理完相關(guān)事情后才會繼續(xù)向下執(zhí)行恩商,確實是以sync(同步)的方式進行的变逃。我們在看BYNotifyReceiver在接收到通知后打印的log日志,可以發(fā)現(xiàn)相關(guān)處理邏輯所在的線程和發(fā)送通知所在的線程是同一個線程怠堪。一句話就是揽乱,Notification是以sync(同步)的方式發(fā)送的,也就是說當(dāng)發(fā)送通知出去以后會先卡住當(dāng)前線程粟矿,等到所有注冊該通知者接收到此通知并在當(dāng)前線程(即發(fā)出通知的線程)進行完相對應(yīng)的處理后凰棉,程序才會繼續(xù)往下執(zhí)行。

注冊通知的另一個方法

我們繼續(xù)往下看陌粹,在注冊通知的API中或許我們還會發(fā)現(xiàn)一個比較有趣的API,如下:

- (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));
    // The return value is retained by the system, and should be held onto by the caller in
    // order to remove the observer with removeObserver: later, to stop observation.

我們可以看到撒犀,此方法在給通知注冊觀察者的時候,加入了queue和block掏秩,簡化了代碼或舞,同樣也使代碼邏輯更加緊湊,代碼的可讀性也更高蒙幻。它并沒有指定observer,而是返回一個observer映凳。在這里,如果指定queue,block的執(zhí)行將在你指定的queue中執(zhí)行邮破,若沒有指定queue诈豌,block的執(zhí)行將在post消息線程中執(zhí)行。注意抒和,此處block會引起循環(huán)引用矫渔,使用過程中要使用weakSelf。關(guān)鍵代碼如下:

//將BYNotifyReceiver中代碼調(diào)整如下
- (instancetype)init
{
    if (self = [super init]) {
   
        NSOperationQueue *queue = [[NSOperationQueue alloc] init];
        queue.name = @"NOTIFICATION_HANDLE_QUEUE";
        __weak typeof(self) weakSelf = self;
       _notificationObserver = [[NSNotificationCenter defaultCenter] addObserverForName:@"BYNotificationTest" object:nil queue:queue usingBlock:^(NSNotification * _Nonnull note) {
           [weakSelf testQueueNotification];
        }];
    }
    return self;
}

- (void)testQueueNotification
{
   [NSThread sleepForTimeInterval:5];//阻塞當(dāng)前線程5秒
    if ([[NSThread currentThread] isMainThread]) {
        NSLog(@"接收到通知 - BYNotificationTest摧莽,主線程庙洼,開始處理相關(guān)邏輯,currentQueue = %@",[NSOperationQueue currentQueue].name);
    } else {
        NSLog(@"接收到通知 - BYNotificationTest,異步線程范嘱,開始處理相關(guān)邏輯,currentQueue = %@",[NSOperationQueue currentQueue].name);
    }
    // 注意移除通知的方式送膳,參數(shù)為notificationObserver,而不是self
    [[NSNotificationCenter defaultCenter] removeObserver:self.notificationObserver];
}

//ViewController中代碼
- (void)viewDidAppear:(BOOL)animated
{
    [super viewDidAppear:animated];
    NSLog(@"------------主線程發(fā)送通知---------------");
    [self.poster postNotification];
    self.receiver = nil;
}

而log日志如下丑蛤,不用過多解釋請仔細看:

2017-12-23 16:37:21.559728+0800 NotificationTest[7607:678009] ------------主線程發(fā)送通知---------------
2017-12-23 16:37:21.559865+0800 NotificationTest[7607:678009] 準備發(fā)送通知 - BYNotificationTest
2017-12-23 16:37:26.565126+0800 NotificationTest[7607:678048] 接收到通知 - BYNotificationTest叠聋,異步線程,開始處理相關(guān)邏輯,currentQueue = NOTIFICATION_HANDLE_QUEUE
2017-12-23 16:37:26.565565+0800 NotificationTest[7607:678009] BYNotificationTest - 通知發(fā)送完畢受裹,繼續(xù)往下執(zhí)行

異步通知的實現(xiàn)

如何實現(xiàn)異步通知呢碌补?要想真正做到async(異步)發(fā)送通知虏束,則必須使用NSNotificationQueue。
一般來講厦章,這個是為了合并多個通知一起發(fā)送時使用的镇匀,當(dāng)然也能用作async發(fā)送通知。其中有幾個比較重要的點袜啃,要知道汗侵。

NSPostingStyle用來決定何時發(fā)送通知。

typedef NS_ENUM(NSUInteger, NSPostingStyle) {
    NSPostWhenIdle = 1,  //當(dāng)前runloop處于空閑或者waiting狀態(tài)時post
    NSPostASAP = 2,      //當(dāng)前runloop結(jié)束時post
    NSPostNow = 3        //立即post群发,使用此模式相當(dāng)于sync!
};

NSNotificationCoalescing決定是否將多個通知進行合成晰韵,然后是以哪種方式合成。

typedef NS_OPTIONS(NSUInteger, NSNotificationCoalescing) {
    NSNotificationNoCoalescing = 0,        //不合并
    NSNotificationCoalescingOnName = 1,    //合并name(名稱)相同的通知
    NSNotificationCoalescingOnSender = 2   //合并sender(發(fā)送者)相同的通知
};

還可以設(shè)定runloop mode熟妓。如果設(shè)定了runloop model則只會在特定的runloop mode下才會進行發(fā)送通知雪猪。

相關(guān)代碼如下:

BYNotifyReceiver.m

- (instancetype)init
{
    if (self = [super init]) {
       [[NSNotificationCenter defaultCenter] addObserver:self selector:@selector(handleAsyncNotification:) name:@"BYAsyncNotificationTest" object:nil];
    }
    return self;
}

- (void)handleAsyncNotification:(NSNotification *)note
{
    if ([[NSThread currentThread] isMainThread]) {
        NSLog(@"接收到通知 - BYAsyncNotificationTest,主線程起愈,開始處理相關(guān)邏輯,currentQueue = %@",[NSOperationQueue currentQueue].name);
    } else {
        NSLog(@"接收到通知 - BYAsyncNotificationTest只恨,異步線程,開始處理相關(guān)邏輯,currentQueue = %@",[NSOperationQueue currentQueue].name);
    }
}

BYNotifyPoster.m

/*!
 * @brief 異步發(fā)送通知
 */
- (void)asyncPostNotification
{
    NSLog(@"準備發(fā)送通知 - BYAsyncNotificationTest");
    NSNotification *asyncNotification = [NSNotification notificationWithName:@"BYAsyncNotificationTest" object:nil];
    [[NSNotificationQueue defaultQueue] enqueueNotification:asyncNotification postingStyle:NSPostASAP];
    NSLog(@"BYAsyncNotificationTest - 通知發(fā)送完畢抬虽,繼續(xù)往下執(zhí)行");
    // TODO:發(fā)送通知完畢后官觅,繼續(xù)執(zhí)行
    [self testAsync];
}

- (void)testAsync
{
    NSLog(@"%@", NSStringFromSelector(_cmd));
}

ViewController.m

- (void)viewDidAppear:(BOOL)animated
{
    [super viewDidAppear:animated];
    NSLog(@"------------主線程發(fā)送通知---------------");
    [self.poster asyncPostNotification];
}

運行后,打印Log如下斥赋,答案一目了然:

2017-12-23 17:58:24.080311+0800 NotificationTest[9651:900465] ------------主線程發(fā)送通知---------------
2017-12-23 17:58:24.080736+0800 NotificationTest[9651:900465] 準備發(fā)送通知 - BYAsyncNotificationTest
2017-12-23 17:58:24.080926+0800 NotificationTest[9651:900465] BYAsyncNotificationTest - 通知發(fā)送完畢缰猴,繼續(xù)往下執(zhí)行
2017-12-23 17:58:24.081192+0800 NotificationTest[9651:900465] testAsync
2017-12-23 17:58:24.081644+0800 NotificationTest[9651:900465] 接收到通知 - BYAsyncNotificationTest,主線程疤剑,開始處理相關(guān)邏輯,currentQueue = NSOperationQueue Main Queue

至于如何合并通知滑绒,這里暫時不再講了,測試代碼就無需上傳了隘膘,留給想玩的同志們?nèi)ネ嬉晒省H绻ㄖ獛в衭serinfo參數(shù),此時通知被合并弯菊,userinfo又如何處理呢纵势,是否也會被合并呢,這個地方是比較有意思的管钳。如果被合并將會是什么樣子呢钦铁?如果不被合并,那通知接受者收到的userinfo中又是什么呢才漆?

本文為原創(chuàng)文章牛曹,轉(zhuǎn)載請注明出處。

可加群一起交流共同學(xué)習(xí):801216530醇滥。

最后編輯于
?著作權(quán)歸作者所有,轉(zhuǎn)載或內(nèi)容合作請聯(lián)系作者
  • 序言:七十年代末黎比,一起剝皮案震驚了整個濱河市超营,隨后出現(xiàn)的幾起案子,更是在濱河造成了極大的恐慌阅虫,老刑警劉巖演闭,帶你破解...
    沈念sama閱讀 211,376評論 6 491
  • 序言:濱河連續(xù)發(fā)生了三起死亡事件,死亡現(xiàn)場離奇詭異颓帝,居然都是意外死亡米碰,警方通過查閱死者的電腦和手機,發(fā)現(xiàn)死者居然都...
    沈念sama閱讀 90,126評論 2 385
  • 文/潘曉璐 我一進店門躲履,熙熙樓的掌柜王于貴愁眉苦臉地迎上來见间,“玉大人聊闯,你說我怎么就攤上這事工猜。” “怎么了菱蔬?”我有些...
    開封第一講書人閱讀 156,966評論 0 347
  • 文/不壞的土叔 我叫張陵篷帅,是天一觀的道長。 經(jīng)常有香客問我拴泌,道長魏身,這世上最難降的妖魔是什么? 我笑而不...
    開封第一講書人閱讀 56,432評論 1 283
  • 正文 為了忘掉前任蚪腐,我火速辦了婚禮箭昵,結(jié)果婚禮上,老公的妹妹穿的比我還像新娘回季。我一直安慰自己家制,他們只是感情好,可當(dāng)我...
    茶點故事閱讀 65,519評論 6 385
  • 文/花漫 我一把揭開白布泡一。 她就那樣靜靜地躺著颤殴,像睡著了一般。 火紅的嫁衣襯著肌膚如雪鼻忠。 梳的紋絲不亂的頭發(fā)上涵但,一...
    開封第一講書人閱讀 49,792評論 1 290
  • 那天,我揣著相機與錄音帖蔓,去河邊找鬼矮瘟。 笑死,一個胖子當(dāng)著我的面吹牛塑娇,可吹牛的內(nèi)容都是我干的澈侠。 我是一名探鬼主播,決...
    沈念sama閱讀 38,933評論 3 406
  • 文/蒼蘭香墨 我猛地睜開眼钝吮,長吁一口氣:“原來是場噩夢啊……” “哼埋涧!你這毒婦竟也來了板辽?” 一聲冷哼從身側(cè)響起,我...
    開封第一講書人閱讀 37,701評論 0 266
  • 序言:老撾萬榮一對情侶失蹤棘催,失蹤者是張志新(化名)和其女友劉穎劲弦,沒想到半個月后,有當(dāng)?shù)厝嗽跇淞掷锇l(fā)現(xiàn)了一具尸體醇坝,經(jīng)...
    沈念sama閱讀 44,143評論 1 303
  • 正文 獨居荒郊野嶺守林人離奇死亡邑跪,尸身上長有42處帶血的膿包…… 初始之章·張勛 以下內(nèi)容為張勛視角 年9月15日...
    茶點故事閱讀 36,488評論 2 327
  • 正文 我和宋清朗相戀三年,在試婚紗的時候發(fā)現(xiàn)自己被綠了呼猪。 大學(xué)時的朋友給我發(fā)了我未婚夫和他白月光在一起吃飯的照片画畅。...
    茶點故事閱讀 38,626評論 1 340
  • 序言:一個原本活蹦亂跳的男人離奇死亡,死狀恐怖宋距,靈堂內(nèi)的尸體忽然破棺而出轴踱,到底是詐尸還是另有隱情,我是刑警寧澤谚赎,帶...
    沈念sama閱讀 34,292評論 4 329
  • 正文 年R本政府宣布淫僻,位于F島的核電站,受9級特大地震影響壶唤,放射性物質(zhì)發(fā)生泄漏雳灵。R本人自食惡果不足惜,卻給世界環(huán)境...
    茶點故事閱讀 39,896評論 3 313
  • 文/蒙蒙 一闸盔、第九天 我趴在偏房一處隱蔽的房頂上張望悯辙。 院中可真熱鬧,春花似錦迎吵、人聲如沸躲撰。這莊子的主人今日做“春日...
    開封第一講書人閱讀 30,742評論 0 21
  • 文/蒼蘭香墨 我抬頭看了看天上的太陽茴肥。三九已至,卻和暖如春荡灾,著一層夾襖步出監(jiān)牢的瞬間瓤狐,已是汗流浹背。 一陣腳步聲響...
    開封第一講書人閱讀 31,977評論 1 265
  • 我被黑心中介騙來泰國打工批幌, 沒想到剛下飛機就差點兒被人妖公主榨干…… 1. 我叫王不留础锐,地道東北人。 一個月前我還...
    沈念sama閱讀 46,324評論 2 360
  • 正文 我出身青樓荧缘,卻偏偏與公主長得像皆警,于是被迫代替她去往敵國和親。 傳聞我的和親對象是個殘疾皇子截粗,可洞房花燭夜當(dāng)晚...
    茶點故事閱讀 43,494評論 2 348

推薦閱讀更多精彩內(nèi)容

  • 設(shè)計模式是什么信姓? 你知道哪些設(shè)計模式鸵隧,并簡要敘述? 設(shè)計模式是一種編碼經(jīng)驗意推,就是用比較成熟的邏輯去處理某一種類型的...
    iOS菜鳥大大閱讀 702評論 0 1
  • 多線程豆瘫、特別是NSOperation 和 GCD 的內(nèi)部原理。運行時機制的原理和運用場景菊值。SDWebImage的原...
    LZM輪回閱讀 2,004評論 0 12
  • 2018年4月19日上午外驱,春光明媚,付草樓鄉(xiāng)全體中小學(xué)校長在芝麻洼中心校 領(lǐng)導(dǎo)王均生 馬守華 李振山 李書博 ...
    竹海聽濤_d5d8閱讀 603評論 0 0
  • 好不容易有了一次緣分 好不容易有了一次回眸 好不容易在熙熙攘攘的人群里撲捉了你的身影 好不容易讓自己有了愛你的一次...
    花的憂傷閱讀 291評論 1 5
  • 目前陌生人社交領(lǐng)域似乎缺少一款搜索功能較強的軟件腻窒,我最近正在構(gòu)思設(shè)計這樣的軟件昵宇,打算盡快把它做出來。有沒有對這個領(lǐng)...
    明銳價值閱讀 348評論 2 3