macOS用戶態(tài)進(jìn)程間的通信實(shí)現(xiàn)

macOS內(nèi)核與用戶態(tài)進(jìn)程的通信前面已經(jīng)講解過(guò)了,用戶態(tài)進(jìn)程間通信較為簡(jiǎn)單耗拓,Apple官方推薦使用XPC機(jī)制表悬。XPC有底層的C API炫贤,也有封裝后的上層的objc和swift API出吹,可以實(shí)現(xiàn)像調(diào)用自身進(jìn)程的接口一樣調(diào)用對(duì)端的暴露接口颤诀。這里以swift為例說(shuō)明XPC的代碼實(shí)現(xiàn)谜慌。開源項(xiàng)目NuwaStone用戶態(tài)進(jìn)程間通信均基于XPC實(shí)現(xiàn)然想,更多代碼細(xì)節(jié)可參考該項(xiàng)目。

對(duì)外接口及類定義

使用XPC需新建兩個(gè)協(xié)議欣范,作為自身進(jìn)程和對(duì)端進(jìn)程的通信接口变泄,然后新建一個(gè)處理XPC連接請(qǐng)求的類令哟,注意需繼承NSObject,以便實(shí)現(xiàn)XPC系統(tǒng)調(diào)用所需的代理接口妨蛹。注意屏富,為了方便XPCConnection類是自身進(jìn)程與對(duì)端進(jìn)程共用的,但部分方法是不能共用的蛙卤,詳細(xì)見注釋狠半。

// 客戶端進(jìn)程協(xié)議
@objc protocol ClientXPCProtocol {
}

// 服務(wù)端進(jìn)程協(xié)議
@objc protocol ServerXPCProtocol {
    func connectResponse(_ handler: @escaping (Bool) -> Void)
}

// 連接請(qǐng)求處理類
class XPCConnection: NSObject {
    static let shared = XPCConnection()
    var listener: NSXPCListener?
    // 連接處理對(duì)象
    var connection: NSXPCConnection?
    
    private func getMachServiceName(from bundle: Bundle) -> String {
        let clientKeys = bundle.object(forInfoDictionaryKey: ClientName) as? [String: Any]
        let machServiceName = clientKeys?[MachServiceKey] as? String
        return machServiceName ?? ""
    }
    
    // 僅服務(wù)端進(jìn)程調(diào)用
    func startListener() {
        let newListener = NSXPCListener(machServiceName: "your service name")
        newListener.delegate = self
        newListener.resume()
        listener = newListener
        Logger(.Info, "Start XPC listener successfully.")
    }
    
    // 僅客戶端進(jìn)程調(diào)用
    func connectToServer(bundle: Bundle, delegate: ClientXPCProtocol, handler: @escaping (Bool) -> Void) {
        guard connection == nil else {
            Logger(.Info, "Client already connected.")
            handler(true)
            return
        }
        guard getMachServiceName(from: bundle) == ClientBundle else {
            handler(false)
            return
        }
        
        let newConnection = NSXPCConnection(machServiceName: DaemonBundle)
        newConnection.exportedObject = delegate
        newConnection.exportedInterface = NSXPCInterface(with: ClientXPCProtocol.self)
        newConnection.remoteObjectInterface = NSXPCInterface(with: DaemonXPCProtocol.self)
        connection = newConnection
        newConnection.resume()
        
        let proxy = newConnection.remoteObjectProxyWithErrorHandler { error in
            Logger(.Error, "Failed to connect with error [\(error)]")
            self.connection?.invalidate()
            self.connection = nil
            handler(false)
        } as? DaemonXPCProtocol
        
        proxy?.connectResponse(handler)
    }
}

XPC的代碼實(shí)現(xiàn)是較為簡(jiǎn)單的,startListener函數(shù)首先需創(chuàng)建連接所用的Mach端口颤难,網(wǎng)絡(luò)資源在Mac內(nèi)核基本抽象為Mach端口資源神年,然后設(shè)置處理連接請(qǐng)求的代理為自身類,后面需編寫請(qǐng)求處理代碼以校驗(yàn)連接并建立連接行嗤。connectToServer函數(shù)首先判斷是否連接已建立已日,然后簡(jiǎn)單校驗(yàn)了一下APP包名。創(chuàng)建連接對(duì)象時(shí)需設(shè)置自身暴露接口和遠(yuǎn)程調(diào)用接口栅屏,設(shè)置完成后所調(diào)用的connectResponse為連接測(cè)試函數(shù)飘千。在調(diào)用遠(yuǎn)程接口前均需獲取remoteObjectProxy遠(yuǎn)程對(duì)象代理,這里需進(jìn)行錯(cuò)誤處理栈雳,如果出錯(cuò)則表示連接斷開占婉。

XPC連接請(qǐng)求系統(tǒng)調(diào)用代理接口

startListener函數(shù)中設(shè)置處理連接請(qǐng)求的代理為自身類,所以類需要實(shí)現(xiàn)代理方法甫恩,代理方法的實(shí)現(xiàn)很簡(jiǎn)單,僅需實(shí)現(xiàn)下面一種方法酌予。函數(shù)內(nèi)的實(shí)現(xiàn)看起來(lái)與connectToServer差不多磺箕,不過(guò)多了錯(cuò)誤處理機(jī)制。建立連接前可以進(jìn)行簽名校驗(yàn)以增強(qiáng)安全性抛虫。這里的invalidationHandler被系統(tǒng)調(diào)用時(shí)表明連接正常斷開松靡,interruptionHandler被系統(tǒng)調(diào)用時(shí)表明連接意外斷開,如果有斷開重連要求可以加在這里建椰。

extension XPCConnection: NSXPCListenerDelegate {
    func listener(_ listener: NSXPCListener, shouldAcceptNewConnection newConnection: NSXPCConnection) -> Bool {
        newConnection.exportedObject = self
        newConnection.exportedInterface = NSXPCInterface(with: DaemonXPCProtocol.self)
        newConnection.remoteObjectInterface = NSXPCInterface(with: ClientXPCProtocol.self)
        newConnection.invalidationHandler = {
            self.connection = nil
            Logger(.Info, "Client disconnected.")
        }
        newConnection.interruptionHandler = {
            self.connection = nil
            Logger(.Info, "Client interrupted.")
        }
        
        connection = newConnection
        newConnection.resume()
        return true
    }

服務(wù)端對(duì)外暴露接口

暴露接口根據(jù)自己的項(xiàng)目需要自行設(shè)計(jì)實(shí)現(xiàn)就好雕欺,但要注意這些接口是無(wú)返回值的,畢竟是遠(yuǎn)程調(diào)用棉姐,基本也不會(huì)需要返回值屠列。這里使用@escaping修飾代碼塊,表示代碼塊的調(diào)用可能在函數(shù)返回后伞矩。

extension XPCConnection: DaemonXPCProtocol {
    func connectResponse(_ handler: @escaping (Bool) -> Void) {
        Logger(.Info, "Client connected.")
        handler(true)
    }
}
最后編輯于
?著作權(quán)歸作者所有,轉(zhuǎn)載或內(nèi)容合作請(qǐng)聯(lián)系作者
  • 序言:七十年代末笛洛,一起剝皮案震驚了整個(gè)濱河市,隨后出現(xiàn)的幾起案子乃坤,更是在濱河造成了極大的恐慌苛让,老刑警劉巖沟蔑,帶你破解...
    沈念sama閱讀 212,029評(píng)論 6 492
  • 序言:濱河連續(xù)發(fā)生了三起死亡事件,死亡現(xiàn)場(chǎng)離奇詭異狱杰,居然都是意外死亡瘦材,警方通過(guò)查閱死者的電腦和手機(jī),發(fā)現(xiàn)死者居然都...
    沈念sama閱讀 90,395評(píng)論 3 385
  • 文/潘曉璐 我一進(jìn)店門仿畸,熙熙樓的掌柜王于貴愁眉苦臉地迎上來(lái)食棕,“玉大人,你說(shuō)我怎么就攤上這事颁湖⌒洌” “怎么了?”我有些...
    開封第一講書人閱讀 157,570評(píng)論 0 348
  • 文/不壞的土叔 我叫張陵甥捺,是天一觀的道長(zhǎng)抢蚀。 經(jīng)常有香客問(wèn)我,道長(zhǎng)镰禾,這世上最難降的妖魔是什么皿曲? 我笑而不...
    開封第一講書人閱讀 56,535評(píng)論 1 284
  • 正文 為了忘掉前任,我火速辦了婚禮吴侦,結(jié)果婚禮上屋休,老公的妹妹穿的比我還像新娘。我一直安慰自己备韧,他們只是感情好劫樟,可當(dāng)我...
    茶點(diǎn)故事閱讀 65,650評(píng)論 6 386
  • 文/花漫 我一把揭開白布。 她就那樣靜靜地躺著织堂,像睡著了一般叠艳。 火紅的嫁衣襯著肌膚如雪。 梳的紋絲不亂的頭發(fā)上易阳,一...
    開封第一講書人閱讀 49,850評(píng)論 1 290
  • 那天附较,我揣著相機(jī)與錄音,去河邊找鬼潦俺。 笑死拒课,一個(gè)胖子當(dāng)著我的面吹牛,可吹牛的內(nèi)容都是我干的事示。 我是一名探鬼主播早像,決...
    沈念sama閱讀 39,006評(píng)論 3 408
  • 文/蒼蘭香墨 我猛地睜開眼,長(zhǎng)吁一口氣:“原來(lái)是場(chǎng)噩夢(mèng)啊……” “哼肖爵!你這毒婦竟也來(lái)了扎酷?” 一聲冷哼從身側(cè)響起,我...
    開封第一講書人閱讀 37,747評(píng)論 0 268
  • 序言:老撾萬(wàn)榮一對(duì)情侶失蹤遏匆,失蹤者是張志新(化名)和其女友劉穎法挨,沒(méi)想到半個(gè)月后谁榜,有當(dāng)?shù)厝嗽跇淞掷锇l(fā)現(xiàn)了一具尸體,經(jīng)...
    沈念sama閱讀 44,207評(píng)論 1 303
  • 正文 獨(dú)居荒郊野嶺守林人離奇死亡凡纳,尸身上長(zhǎng)有42處帶血的膿包…… 初始之章·張勛 以下內(nèi)容為張勛視角 年9月15日...
    茶點(diǎn)故事閱讀 36,536評(píng)論 2 327
  • 正文 我和宋清朗相戀三年窃植,在試婚紗的時(shí)候發(fā)現(xiàn)自己被綠了。 大學(xué)時(shí)的朋友給我發(fā)了我未婚夫和他白月光在一起吃飯的照片荐糜。...
    茶點(diǎn)故事閱讀 38,683評(píng)論 1 341
  • 序言:一個(gè)原本活蹦亂跳的男人離奇死亡巷怜,死狀恐怖,靈堂內(nèi)的尸體忽然破棺而出暴氏,到底是詐尸還是另有隱情延塑,我是刑警寧澤,帶...
    沈念sama閱讀 34,342評(píng)論 4 330
  • 正文 年R本政府宣布答渔,位于F島的核電站关带,受9級(jí)特大地震影響,放射性物質(zhì)發(fā)生泄漏沼撕。R本人自食惡果不足惜宋雏,卻給世界環(huán)境...
    茶點(diǎn)故事閱讀 39,964評(píng)論 3 315
  • 文/蒙蒙 一、第九天 我趴在偏房一處隱蔽的房頂上張望务豺。 院中可真熱鬧磨总,春花似錦、人聲如沸笼沥。這莊子的主人今日做“春日...
    開封第一講書人閱讀 30,772評(píng)論 0 21
  • 文/蒼蘭香墨 我抬頭看了看天上的太陽(yáng)奔浅。三九已至邻薯,卻和暖如春,著一層夾襖步出監(jiān)牢的瞬間乘凸,已是汗流浹背。 一陣腳步聲響...
    開封第一講書人閱讀 32,004評(píng)論 1 266
  • 我被黑心中介騙來(lái)泰國(guó)打工累榜, 沒(méi)想到剛下飛機(jī)就差點(diǎn)兒被人妖公主榨干…… 1. 我叫王不留营勤,地道東北人。 一個(gè)月前我還...
    沈念sama閱讀 46,401評(píng)論 2 360
  • 正文 我出身青樓壹罚,卻偏偏與公主長(zhǎng)得像葛作,于是被迫代替她去往敵國(guó)和親。 傳聞我的和親對(duì)象是個(gè)殘疾皇子猖凛,可洞房花燭夜當(dāng)晚...
    茶點(diǎn)故事閱讀 43,566評(píng)論 2 349

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