欧美bbbwbbbw肥妇,免费乱码人妻系列日韩,一级黄片

NotificationCenter類實現(xiàn)原理

 更新時間:2023年03月31日 08:43:00   作者:向輝_  
這篇文章主要為大家介紹了NotificationCenter類實現(xiàn)原理源碼解析,有需要的朋友可以借鑒參考下,希望能夠有所幫助,祝大家多多進步,早日升職加薪

正文

NotificationCenter是一個系統(tǒng)組件,它負責協(xié)調(diào)和管理事件的通知和響應。它的基本原理是基于觀察者模式!而 Apple 對其是閉源的,因此無法查看 NotificationCenter 的源碼,但是可以通過分析開源的 Swift 來理解 NotificationCenter 的實現(xiàn),以下是一個簡化的實現(xiàn):

簡單實現(xiàn)

1、首先定義一個NotificationCenter類定義

class RYNotificationCenter {
    private init(){}
    static let `default` = RYNotificationCenter()
    private var observers: [RYNotificationObserver] = []
}

定義了一個單例,用于在整個程序中共享,observers數(shù)組用來存儲已經(jīng)注冊的所有觀察者。

2、然后定義一個觀察者對象

觀察者對象用來封裝具體的觀察者的信息。

class RYNotificationObserver {
    var name: String
    var block: (Notification) -> Void
    init(name: String, block: @escaping (Notification) -> Void) {
        self.name = name
        self.block = block
    }
}

3、在NotificationCenter中添加注冊觀察者的方法

func addObserver(name: String, block: @escaping (Notification) -> Void) -> RYNotificationObserver {
    let observer = RYNotificationObserver(name: name, block: block)
    observers.append(observer)
    return observer
}

addObserver方法用于注冊觀察者。在這個實現(xiàn)中,我們創(chuàng)建了一個新 RYNotificationObserver 對象并將其添加到 observers 數(shù)組。這個方法返回觀察者對象,以便稍后從 NotificationCenter 中移除。

4、在 NotificationCenter 中添加發(fā)送通知的方法

/// 發(fā)送通知的本質(zhì)是利用了觀察者模式
/// 讓觀察者數(shù)組執(zhí)行閉包中的代碼
func post(name: String, userInfo: [AnyHashable: Any]? = nil) {
    let notification = Notification(name: Notification.Name(name), userInfo: userInfo)
    observers
        .filter({ $0.name == name })
        .forEach { $0.block(notification) }
}

post 方法用來發(fā)送通知,它接受通知名以及可選的userInfo字典。同時參數(shù)都包裝在Notification對象中,然后遍歷 observers 數(shù)組。如果觀察者的名稱和通知名稱匹配,我們將執(zhí)行保存的block。

5、在NotificationCenter中添加移除通知者的方法

func removeObserver(_ observer: RYNotificationObserver) {
    if let index = observers.firstIndex(where: { $0 === observer }) {
        observers.remove(at: index)
    }
}

removeObserver 方法用于移除觀察者。它接受一個觀察者對象并從 observers 數(shù)組中移除它。

NotificationCenter的源碼分析

普遍來說,現(xiàn)在分析 NotificationCenter 的源碼,一般是 github.com/gnustep/lib… ,這是在 gnustep 庫的源碼中,它和官方的具體實現(xiàn)肯定是有差異的,但是可以以它為參考的對象,在這里通知的源碼使用了三個主要的類:

  • NSNotification
  • NSNotificationCenter
  • NSNotificationQueue

NSNotificationCenter 實現(xiàn)

用于在觀察者和發(fā)送者之間發(fā)送通知,這是核心類,它的方法和Objective-C是一致的,使用 **addObserver:selector:name:object: 方法來添加觀察者,但是它在內(nèi)部使用了C語言實現(xiàn)鏈表的數(shù)據(jù)結構 Obs 存儲觀察者相關的信息:

typedef struct Obs {
  id observer;     /* Object to receive message. */
  SEL selector;     /* Method selector. */
  struct Obs *next; /* Next item in linked list. */
  int retained;     /* Retain count for structure. */
  struct NCTbl *link; /* Pointer back to chunk table */
} Observation;

而在 postNotificationName:object:userInfo: 方法執(zhí)行的時候會通過通知名找到封裝好的 Obs 觀察者,然后執(zhí)行相應的方法:

- (void) postNotificationName: (NSString*)name
      object: (id)object
    userInfo: (NSDictionary*)info
{
  // 先封裝好notification
  GSNotification *notification;
  notification = (id)NSAllocateObject(concrete, 0, NSDefaultMallocZone());
  notification->_name = [name copyWithZone: [self zone]];
  notification->_object = [object retain];
  notification->_info = [info retain];
  [self _postAndRelease: notification];
}
// 然后調(diào)用觀察者的selector方法
- (void) _postAndRealse: (NSNotification*)notification {
......
[o->observer performSelector: o->selector withObject: notification];
......
}

當然,要將封裝好的 notification ,作為參數(shù)傳遞給觀察者需要執(zhí)行的 selector 。

NSNotification 實現(xiàn)

那么 Notifiation 呢?它是一個包含了通知的名稱、發(fā)送者對象以及用戶信息字典的不可變對象。

- (id) initWithCoder: (NSCoder*)aCoder
{
  NSString *name;
  id object;
  NSDictionary *info;
  id n;
  [aCoder decodeValueOfObjCType: @encode(id) at: &name];
  [aCoder decodeValueOfObjCType: @encode(id) at: &object];
  [aCoder decodeValueOfObjCType: @encode(id) at: &info];
  n = [NSNotification notificationWithName: name object: object userInfo: info];
  RELEASE(name);
  RELEASE(object);
  RELEASE(info);
  DESTROY(self);
  return RETAIN(n);
}

NSNotificationQueue 的實現(xiàn)

最后是 NSNotificationQueue 的實現(xiàn),它是一個用于管理通知發(fā)送的隊列,可以按照特定的發(fā)送模式(例如合并相同的通知或按發(fā)送順序)將通知排隊。

- (void) enqueueNotification: (NSNotification*)notification postingStyle:(NSPostingStyle)postingStyle coalesceMask: (NSUInteger)coalesceMask forModes: (NSArray*)modes
{
    if (modes == nil)
    {
      modes = defaultMode;
    }
    if (coalesceMask != NSNotificationNoCoalescing)
    {
      [self dequeueNotificationsMatching: notification coalesceMask: coalesceMask];
    }
    switch (postingStyle) {
        case NSPostNow: {
            NSString *mode;
            mode = [[NSRunLoop currentRunLoop] currentMode];
            if (mode == nil || [modes indexOfObject: mode] != NSNotFound)
            {
                [_center postNotification: notification];
            }
        }
        break;
      case NSPostASAP:
            add_to_queue(_asapQueue, notification, modes, _zone);
            break;
      case NSPostWhenIdle:
            add_to_queue(_idleQueue, notification, modes, _zone);
            break;
    }
}

當使用 NSNotificationQueue 的時候,就不需要我們手動發(fā)送 Notification 了,NSNotificationQueue 會自動幫我們發(fā)送,在上述代碼中,如果是 NSPostNow,那么通知會立馬被發(fā)送,否則就先加入隊列中:_asapQueue 或者 _idleQueue ,然后在合適的時候執(zhí)行隊列中的通知,比如:

void GSPrivateNotifyIdle(NSString *mode) {
    NotificationQueueList *item;
    for (item = currentList(); item; item = item->next)
    {
        if (item->queue) {
            notify(item->queue->_center,
                   item->queue->_idleQueue,
                   mode,
                   item->queue->_zone);
        }
    }
}

問題:如果NotificationCenter 添加的觀察者是self,會造成循環(huán)引用嗎?

答案是:不會!

NotificationCenter 對觀察者的引用方式是弱引用(weak),而不是強持有(strong)。因此,當一個對象被銷毀時,它的 deinit 方法會被調(diào)用,即使它是一個觀察者。所以即使我們不在 deinit 方法中添加移除 self 的操作也是可以的,因為 NotificationCenter 并沒有對觀察者強持有。

問題:如果 NotificationCenter 添加的是 block ,而 block 強持有了 self ,這會造成循環(huán)引用嗎?

答案是:會!

從iOS 9開始,如果使用了基于 block 的觀察者,那么就需要去小心觀察者的生命周期了,因為NotificationCenter 對添加的 block 是強持有的,正如上述簡單實現(xiàn)中的那樣,它對閉包中捕獲的變量就也是強持有的,所以為了避免這種現(xiàn)象,需要確保使用 [weak self] 來捕獲列表。

在實際使用的時候,由于編碼慣性,可能會在 deinit 方法中移除基于 block 的觀察者以解決該問題:

class ViewController: UIViewController {
    private weak var observer: NSObjectProtocol!
    func addObserver() {
        observer = NotificationCenter.default.addObserver(forName: NSNotification.Name("test"), object: nil, queue: OperationQueue.main) { _ in
            self.view.backgroundColor = UIColor.white
        }
    }
    deinit {
        NotificationCenter.default.removeObserver(observer!)
    }
}

但是在這種情況下, deinit 方法并不會執(zhí)行! 原因就是 NotificationCenter 持有了 block, 也間接持有了 self,而 NotificationCenter 是一個單例,所以這種持有關系是一直存在的,導致了 deinit 方法并不會執(zhí)行!

問題:觀察者的 selector 執(zhí)行的線程和發(fā)送通知的線程有關嗎?

答案是:正相關!

從上文中的簡單實現(xiàn)以及GNU的源碼中基本可以看出結論了。添加觀察者的線程并沒有什么影響,而發(fā)送通知的線程,其實就是調(diào)用方法執(zhí)行的線程,所以兩者是在同一線程執(zhí)行的。

func addObserver() {
    NotificationCenter.default.addObserver(self, selector: #selector(click), name: NSNotification.Name.init("test"), object: nil)
    DispatchQueue.global().async {
        NotificationCenter.default.post(name: NSNotification.Name.init("test"), object: nil)
        NSLog("curretThread1: \(Thread.current)")
    }
}
@objc func click() {
    NSLog("curretThread2: \(Thread.current)")
}
// curretThread2: <NSThread: 0x600001358240>{number = 6, name = (null)}
// curretThread1: <NSThread: 0x600001358240>{number = 6, name = (null)}

同時還需要注意的就是通知發(fā)送,然后 selector 被執(zhí)行,這個過程其實本質(zhì)上是一個觀察者模式的實現(xiàn)方式,同時,它也是同步執(zhí)行的,再執(zhí)行完發(fā)送消息的方法后就會去尋找對應的 Observer ,找到之后就執(zhí)行相應的 selector ,執(zhí)行完之后,發(fā)送消息的方法才執(zhí)行完畢了。

所以發(fā)送通知和監(jiān)聽通知執(zhí)行方法的核心是:相同線程執(zhí)行 且 同步執(zhí)行。

以上就是NotificationCenter類實現(xiàn)原理的詳細內(nèi)容,更多關于NotificationCenter 原理的資料請關注腳本之家其它相關文章!

相關文章

  • 深入解析Swift編程中枚舉類型的相關使用

    深入解析Swift編程中枚舉類型的相關使用

    這篇文章主要介紹了Swift編程中枚舉類型的相關使用,是Swift入門學習中的基礎知識,需要的朋友可以參考下
    2015-11-11
  • Swift類型創(chuàng)建之自定義一個類型詳解

    Swift類型創(chuàng)建之自定義一個類型詳解

    這篇文章主要介紹了Swift類型創(chuàng)建之自定義一個類型詳解,本文講解了自定義原型、實現(xiàn)默認值、支持基本布爾型初始化、支持Bool類型判斷、支持兼容各們各派的類型、完善OCBool的布爾基因體系等內(nèi)容,需要的朋友可以參考下
    2015-05-05
  • Swift設計思想Result<T>與Result<T,?E:?Error>類型解析

    Swift設計思想Result<T>與Result<T,?E:?Error>類型解析

    這篇文章主要為大家介紹了Swift設計思想Result<T>與Result<T,?E:?Error>的類型示例解析,有需要的朋友可以借鑒參考下,希望能夠有所幫助,祝大家多多進步,早日升職加薪
    2022-11-11
  • Swift如何調(diào)用Objective-C的可變參數(shù)函數(shù)詳解

    Swift如何調(diào)用Objective-C的可變參數(shù)函數(shù)詳解

    這篇文章主要給大家介紹了關于Swift如何調(diào)用Objective-C的可變參數(shù)函數(shù)的相關資料,文中通過示例代碼介紹的非常詳細,對大家學習或者使用swift具有一定的參考學習價值,需要的朋友們下面隨著小編來一起學習學習吧。
    2018-03-03
  • 關于swift的個人小結

    關于swift的個人小結

    本文是個人對于目前學習swift的一些心得的匯總,這里分享給大家,希望大家能夠喜歡
    2016-12-12
  • Ubuntu 16.04上安裝 Swift 3.0及問題解答

    Ubuntu 16.04上安裝 Swift 3.0及問題解答

    本文給大家分享的是在Ubuntu系統(tǒng)中安裝 Swift 3.0的方法和步驟,以及安裝過程中有可能遇到的問題的解答,這里推薦給小伙伴們,希望大家能夠喜歡
    2016-07-07
  • Swift仿微信語音通話最小化時后的效果實例代碼

    Swift仿微信語音通話最小化時后的效果實例代碼

    這篇文章主要介紹了Swift仿微信語音通話最小化時后的效果的相關資料,文中通過示例代碼介紹的非常詳細,對大家的學習或者工作具有一定的參考學習價值,需要的朋友們下面隨著小編來一起學習學習吧
    2021-03-03
  • Swift實現(xiàn)表格視圖單元格單選(1)

    Swift實現(xiàn)表格視圖單元格單選(1)

    這篇文章主要為大家詳細介紹了Swift實現(xiàn)表格視圖單元格單選,文中示例代碼介紹的非常詳細,具有一定的參考價值,感興趣的小伙伴們可以參考一下
    2022-01-01
  • Swift編程中的初始化與反初始化完全講解

    Swift編程中的初始化與反初始化完全講解

    這篇文章主要介紹了Swift編程中的初始化與反初始化完全講解,是Swift入門學習中的基礎知識,需要的朋友可以參考下
    2015-11-11
  • Swift 4.0中如何引用3.0的第三方庫

    Swift 4.0中如何引用3.0的第三方庫

    這篇文章主要給大家介紹了關于在Swift 4.0中如何引用3.0第三方庫的相關資料,文中通過圖文介紹的非常詳細,對大家的學習或者工作具有一定的參考學習價值,需要的朋友們下面隨著小編來一起學習學習吧。
    2018-01-01

最新評論