溫馨提示×

您好,登錄后才能下訂單哦!

密碼登錄×
登錄注冊(cè)×
其他方式登錄
點(diǎn)擊 登錄注冊(cè) 即表示同意《億速云用戶服務(wù)條款》

iOS中怎么實(shí)現(xiàn)跨頁(yè)面狀態(tài)同步

發(fā)布時(shí)間:2021-06-16 14:31:03 來(lái)源:億速云 閱讀:325 作者:Leah 欄目:移動(dòng)開(kāi)發(fā)

本篇文章為大家展示了iOS中怎么實(shí)現(xiàn)跨頁(yè)面狀態(tài)同步,內(nèi)容簡(jiǎn)明扼要并且容易理解,絕對(duì)能使你眼前一亮,通過(guò)這篇文章的詳細(xì)介紹希望你能有所收獲。

NotificationCenter

狀態(tài)同步實(shí)際是一對(duì)多的場(chǎng)景,也就是一個(gè)事件可以被多個(gè)觀察者監(jiān)聽(tīng)到。而蘋(píng)果的系統(tǒng)框架自帶的 NotificationCenter 正是用來(lái)適配這種場(chǎng)景,并且其也是被系統(tǒng)框架本身及我們開(kāi)發(fā)者大面積使用的。用法如下:

  1. 定義通知名字,以及需要額外傳遞信息的 key

  2. 基于 target-action 的方式注冊(cè)通知

open func addObserver(_ observer: Any, selector aSelector: Selector, name aName: NSNotification.Name?, object anObject: Any?)

實(shí)現(xiàn)監(jiān)聽(tīng)通知的方法

func onReceivedNotification(note: NSNotification)

發(fā)送通知,可以傳遞發(fā)送通知的對(duì)象(object)以及一些額外的信息(userInfo)

open func post(name aName: NSNotification.Name, object anObject: Any?, userInfo aUserInfo: [AnyHashable : Any]? = nil)

移除注冊(cè)的通知

open func removeObserver(_ observer: Any, name aName: NSNotification.Name?, object anObject: Any?)

當(dāng)然 NotificationCenter 也提供了一種更加便利基于 block 的方式注冊(cè)監(jiān)聽(tīng)通知,其將 2,3 兩個(gè)步驟整合為 1 個(gè)步驟。

open func addObserver(forName name: NSNotification.Name?, object obj: Any?, queue: OperationQueue?, using block: @escaping (Notification) -> Void) -> NSObjectProtocol

整體流程很清晰,簡(jiǎn)單易用,但是卻有一個(gè)嚴(yán)重的缺點(diǎn) —— 弱類型。我們接收到的是一個(gè)NSNotification對(duì)象。

open class NSNotification : NSObject, NSCopying, NSCoding {
  open var name: NSNotification.Name { get }
  open var object: Any? { get }
  open var userInfo: [AnyHashable : Any]? { get }
}

假設(shè)我們需要傳遞一個(gè)關(guān)注狀態(tài)改變的信息,那么需要包含關(guān)注更改后的狀態(tài)以及被關(guān)注者的 ID。那么我們需要從 userInfo 中取出所需要的值:

let following = notification.userInfo["FollowingKey"] as! NSNumber
let userID = notification.userInfo["UserIDKey"] as! NSNumber;

也就是說(shuō)接收通知的一方一般需要要查看文檔才知道怎樣從 userInfo 取值,取的值的類型又是什么。這對(duì)于使用是極為不方便的。

SwiftNotificationCenter

SwiftNotificationCenter是一種面向協(xié)議的通知中心方案。使用方式如下:

定義協(xié)議

protocol FollowingChanged {
  func followingDidChange(following: Bool, userID: NSNumber)
}

基于協(xié)議注冊(cè)通知

Broadcaster.register(Update.self, observer: observer)

實(shí)現(xiàn)協(xié)議方法

extension ViewController: FollowingChanged {
 func followingDidChange(following: Bool, userID: NSNumber) {
  // do something
 }
}

發(fā)送通知

Broadcaster.notify(FollowingChanged.self) {
  $0.followingDidChange(following, userID)
}

移除注冊(cè)的通知

Broadcaster.unregister(Update.self, observer: observer)

我們可以看到,其基于協(xié)議的方式解決了弱類型的問(wèn)題,并且其通過(guò)AssociatedObject實(shí)現(xiàn)了通知的自動(dòng)移除。但其也存在著擴(kuò)展性較差的問(wèn)題。

依然是關(guān)注改變的場(chǎng)景,假如隨著業(yè)務(wù)的發(fā)展,有的地方需要知道關(guān)注后是否為互關(guān)的狀態(tài),那么又需要增加一個(gè)字段來(lái)標(biāo)識(shí)。因此我們需要修改協(xié)議,增加參數(shù),且由于其不是必須傳遞的參數(shù),因此是 optional 類型。

protocol FollowingChanging {
  func followingDidChange(following: Bool, userID: NSNumber, followingEachOther: NSNumber?)
}

如果在該類型通知被廣泛應(yīng)用的場(chǎng)景,那么需要修改的地方就尤其多了。這顯然也是難以接受的。

EventBus

EventBus 在安卓中被廣泛地應(yīng)用,其流程如下圖所示:

iOS中怎么實(shí)現(xiàn)跨頁(yè)面狀態(tài)同步

圖片來(lái)源:EventBus

使用方式如下:

定義事件

 class TPFollowingChangedEvent: NSObject, TPEvent {
  	private(set) var following: Bool
  		private(set) var userID: NSNumber
}

注冊(cè)事件

TPEventBus<TPFollowingChangedEvent>.shared.register(eventType: TPFollowingChangedEvent.self, subscriber: self, selector: #selector(onEvent(event:object:)))

實(shí)現(xiàn)監(jiān)聽(tīng)事件的方法

@objc func onEvent(event: TPFollowingChangedEvent, object: Any?) {
  	// do something
}

發(fā)送事件

TPEventBus.shared.post(event: event, object: self)

移除事件的注冊(cè)

TPEventBus<TPFollowingChangedEvent>.shared.unregister(eventType: TPFollowingChangedEvent.self, subscriber: self)

我們可以看到, EventBus 也是強(qiáng)類型的。

假如依然關(guān)注的場(chǎng)景,需要增加 followingEachOther 參數(shù),那么我們只需要在 TPFollowingChangedEvent 中增加 followingEachOther 參數(shù)即可。如下所示:

class TPFollowingChangedEvent: NSObject, TPEvent {
  	private(set) var following: Bool
  		private(set) var userID: NSNumber
  	private(set) var followingEachOther: NSNumber?
}

因此使用 EventBus 實(shí)現(xiàn)了以下需求:

  • 強(qiáng)類型

  • 可擴(kuò)展

EventBus 同 NotificationCenter 都是基于 target-action 的方案,但是我們不難將其擴(kuò)展為支持 block 監(jiān)聽(tīng)的方式,并且同樣讓其能夠自動(dòng)移除事件的注冊(cè)。類似于如下的使用方式:

TPEventBus<TPFollowingChangedEvent>.shared.subscribe(eventType: TPFollowingChangedEvent.self).forObject(self).onQueue(OperationQueue.main).onEvent { (event, object) in
   // do something
}.disposed(by: self)

上述內(nèi)容就是iOS中怎么實(shí)現(xiàn)跨頁(yè)面狀態(tài)同步,你們學(xué)到知識(shí)或技能了嗎?如果還想學(xué)到更多技能或者豐富自己的知識(shí)儲(chǔ)備,歡迎關(guān)注億速云行業(yè)資訊頻道。

向AI問(wèn)一下細(xì)節(jié)

免責(zé)聲明:本站發(fā)布的內(nèi)容(圖片、視頻和文字)以原創(chuàng)、轉(zhuǎn)載和分享為主,文章觀點(diǎn)不代表本網(wǎng)站立場(chǎng),如果涉及侵權(quán)請(qǐng)聯(lián)系站長(zhǎng)郵箱:is@yisu.com進(jìn)行舉報(bào),并提供相關(guān)證據(jù),一經(jīng)查實(shí),將立刻刪除涉嫌侵權(quán)內(nèi)容。

ios
AI