如題,今天的博客咱們就來記錄一下iOS開發中使用MachPort來實現線程間的通訊,而後使用該知識點來轉發子線程中所發出的Notification。簡單的說,MachPort的工做方式實際上是將NSMachPort的對象添加到一個線程所對應的RunLoop中,並給NSMachPort對象設置相應的代理。在其餘線程中調用該MachPort對象發消息時會在MachPort所關聯的線程中執行相關的代理方法。html
下方內容咱們先來看一下MachPort的工做方式,而後再看一下在子線程中發Notification的效果,最後咱們在經過MachPort來說子線程中的發出的通知轉發到主線程中進行處理。git
1、MachPort的使用方式github
接下來咱們就經過一個小的示例來簡單的看一下MachPort的使用方式。首先咱們聲明瞭一個NSMachPort的成員屬性handelEventMachPort,該變量實例化後指定其NSMachPortDelegate的對象爲當前類。而後將handelEventMachPort添加到主線程中,具體代碼以下所示。app
搞定NSMachPort對象後,接下來咱們要在當前VC實現NSMachPortDelegate代理中相關的方法,以下所示。當在其餘線程中調用上述的MachPort對象發送消息時,會在主線程中執行下方的代理方法。在該方法中咱們打印了該方法執行時所在的線程,具體代碼以下所示:異步
實例化完MachPort對象以及實現其相關的代理方法後,接下來要作的事情就是開闢一個新的線程,而後在這個新的線程中調用handelEventMachPort對象,往主線程所對應的RunLoop中發送消息。 oop
代碼實現完畢後,接下來就該看一下運行效果了。下方就是上述代碼示例所運行的結果。從結果中咱們不難看出,點擊按鈕時,會開啓一個新的子線程,咱們將這個開啓的子線程命名爲「MySubThread」。在這個子線程中咱們調用了與主線程關聯的MachPort對象發送消息。而後在主線程中執行該MachPort對象的相關回調方法,每次點擊按鈕的輸出以下所示:ui
2、子線程中Notification的發送spa
該部分算是爲下一部分作鋪墊的,本部分的代碼示例比較簡單。作的事情主要是在主線程中註冊一個觀察者,而後在開啓的子線程中發送通知,咱們來看一下處理該通知的方法所處的線程。線程
下方就是本部分的核心代碼,代碼比較簡單。首先咱們打印出註冊觀察者的線程,而後往通知中心添加觀察者。緊接着咱們就建立一個子線程,而後對子線程的信息進行打印並獲取通知中心單例發送通知。3d
而後在收到通知事件所執行的方法中,咱們要作的事情就是對執行該方法的線程進行打印。具體代碼以下所示:
實現完上述代碼後,下方是上述代碼的運行結果。從結果中咱們不難發現,雖然是在主線程中添加的觀察者,可是若是在子線程中發出通知,那麼就在該子線程中處理通知所關聯的方法,具體效果以下所示:
3、將子線程發出的通知經過MachPort轉發到主線程中進行處理
接下來所作的事情就是將第一部分和第二部分的內容進行整合。也就是將子線程發出的通知經過MachPort轉發到主線程中進行處理。下方的代碼示例咱們參考了Apple Developer中的相關示例(連接請戳我)。固然了,對其官方示例咱們作了一些修改,目的是爲了更易於理解。
首先仍是得實現NSMachPortDelegate相關協議中的方法,下方代碼段中的notificationQueue用來純粹子線程發出的全部通知,mainThread則是用來儲存主線程了,lock則是對通知隊列加鎖,避免多個線程同時操做該隊列所出現的數據不一致問題。mackPort則是用於向指望線程發送信號的通訊端口。
下方的代碼段則是對上述字段的賦值。
接着咱們在viewDidLoad方法中打印了註冊通知的線程,固然此處是主線程了。而後在子線程中異步的發送一條通知,具體代碼以下所示:
下方就是收到通知後所執行的方法,在該方法中,咱們看到作了一個判斷。若是該方法是在咱們預期的主線程中被執行的話,那麼咱們就執行收到通知後所要執行的任務。若是不是咱們預期的主線程的話,接下來走的就是經過MachPort來轉發到主線程了。
在轉發通知前要把當前方法所接收到的notification入隊列暫存,等轉發後,在MachPort的相關代理方法中取出相關的通知並作相關處理。
下方代碼段就是處理MachPort所轉發過來的消息。在該方法中取出了隊列中暫存的相關通知並進行了相關處理。代碼以下所示。
下方是具體的運行結果:
本篇博客所涉及demo在github上的分享地址以下: