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

談談iOS中的多繼承與多重代理

 更新時間:2018年10月21日 14:30:40   作者:Dariel  
這篇文章主要給大家介紹了關于iOS中多繼承與多重代理的相關資料,文中通過示例代碼介紹的非常詳細,對大家的學習或者工作具有一定的參考學習價值,需要的朋友們下面隨著小編來一起學習學習吧

前言

多繼承和多重代理在swift的語言層面上是不支持的,但我們有時會遇到這樣的問題:

  • 類B和C分別繼承自A,B1和B2繼承自B,C1和C2繼承自C.現(xiàn)在我們需要在B1和C1中添加相同的方法,怎么去做?使用繼承的話只能在類A中添加,但這樣做的結果是基類A會越來越臃腫,最后變成上帝類God Class,維護起來會很困難.
  • 在實現(xiàn)完某個代理后發(fā)現(xiàn),我們還要在其他頁面中獲取數(shù)據(jù).例如,IM消息接收之后要在多個地方做回調,比如顯示消息內(nèi)容頁面,改變小紅點,顯示消息數(shù).即一對多的模式,我們第一反應是用通知,但通知還是能少用就少用,用多了代碼的可閱讀性會大大降低.

面對第一種情況,最好的解決方法是,B1和C1的公共方法專門封裝到一個地方,需要的時候就調用一下,多繼承就是一個最好的解決方案.

1. 多繼承

1. 實現(xiàn)過程

swift中的類可以遵守多個協(xié)議,但是只可以繼承一個類,而值類型(結構體和枚舉)只能遵守單個或多個協(xié)議,不能做繼承操作.

多繼承的實現(xiàn):協(xié)議的方法可以在該協(xié)議的extension中實現(xiàn)

protocol Behavior {
 func run()
}
extension Behavior {
 func run() {
  print("Running...")
 }
}

struct Dog: Behavior {}

let myDog = Dog()
myDog.run() // Running...

無論是結構體還是類還是枚舉都可以遵守多個協(xié)議,所以要實現(xiàn)多繼承,無非就是多遵守幾個協(xié)議的問題.

下面舉個例子.

2. 通過多繼承為UIView擴展方法

// MARK: - 閃爍功能
protocol Blinkable {
 func blink()
}
extension Blinkable where Self: UIView {
 func blink() {
  alpha = 1
  
  UIView.animate(
   withDuration: 0.5,
   delay: 0.25,
   options: [.repeat, .autoreverse],
   animations: {
    self.alpha = 0
  })
 }
}

// MARK: - 放大和縮小
protocol Scalable {
 func scale()
}
extension Scalable where Self: UIView {
 func scale() {
  transform = .identity
  
  UIView.animate(
   withDuration: 0.5,
   delay: 0.25,
   options: [.repeat, .autoreverse],
   animations: {
    self.transform = CGAffineTransform(scaleX: 1.5, y: 1.5)
  })
 }
}

// MARK: - 添加圓角
protocol CornersRoundable {
 func roundCorners()
}
extension CornersRoundable where Self: UIView {
 func roundCorners() {
  layer.cornerRadius = bounds.width * 0.1
  layer.masksToBounds = true
 }
}

extension UIView: Scalable, Blinkable, CornersRoundable {}

 cyanView.blink()
 cyanView.scale()
 cyanView.roundCorners()


這樣,如果我們自定義了其他View,只需要放大和縮小效果,遵守Scalable協(xié)議就可以啦!

3. 多繼承鉆石問題(Diamond Problem),及解決辦法

請看下面代碼

protocol ProtocolA {
  func method()
}

extension ProtocolA {
  func method() {
    print("Method from ProtocolA")
  }
}

protocol ProtocolB {
  func method()
}

extension ProtocolB {
  func method() {
    print("Method from ProtocolB")
  }
}

class MyClass: ProtocolA, ProtocolB {}

此時ProtocolA和ProtocolB都有一個默認的實現(xiàn)方法method(),由于編譯器不知道繼承過來的method()方法是哪個,就會報錯.

💎鉆石問題Diamond Problem,當某一個類或值類型在繼承圖譜中有多條路徑時就會發(fā)生.

解決方法:

1. 在目標值類型或類中重寫那個發(fā)生沖突的方法method().

2. 直接修改協(xié)議中重復的方法.

文章開頭我們提到的問題2,我們可以試著用多重代理去解決這個問題.

2. 多重代理

1. 多重代理的實現(xiàn)過程

我們以一個代理的經(jīng)典問題來表述:

主人叫寵物們?nèi)コ燥?吃這個動作作為一個協(xié)議,我們要做到統(tǒng)一管理.

1. 定義協(xié)議

protocol MasterOrderDelegate: class {
  func toEat(_ food: String)
}

2. 定義一個類: 用來管理遵守協(xié)議的類

這邊用了NSHashTable來存儲遵守協(xié)議的類,NSHashTable和NSSet類似,但又有所不同,總的來說有這幾個特點:

1. NSHashTable中的元素可以通過Hashable協(xié)議來判斷是否相等.

2. NSHashTable中的元素如果是弱引用,對象銷毀后會被移除,可以避免循環(huán)引用.

class masterOrderDelegateManager : MasterOrderDelegate {
  private let multiDelegate: NSHashTable<AnyObject> = NSHashTable.weakObjects()

  init(_ delegates: [MasterOrderDelegate]) {
    delegates.forEach(multiDelegate.add)
  }
  
  // 協(xié)議中的方法,可以有多個
  func toEat(_ food: String) {
    invoke { $0.toEat(food) }
  }
  
  // 添加遵守協(xié)議的類
  func add(_ delegate: MasterOrderDelegate) {
    multiDelegate.add(delegate)
  }
  
  // 刪除指定遵守協(xié)議的類
  func remove(_ delegateToRemove: MasterOrderDelegate) {
    invoke {
      if $0 === delegateToRemove as AnyObject {
        multiDelegate.remove($0)
      }
    }
  }
  
  // 刪除所有遵守協(xié)議的類
  func removeAll() {
    multiDelegate.removeAllObjects()
  }

  // 遍歷所有遵守協(xié)議的類
  private func invoke(_ invocation: (MasterOrderDelegate) -> Void) {
    for delegate in multiDelegate.allObjects.reversed() {
      invocation(delegate as! MasterOrderDelegate)
    }
  }
}

3. 其余部分

class Master {
  weak var delegate: MasterOrderDelegate?
  func orderToEat() {
    delegate?.toEat("meat")
  }
}

class Dog {}
extension Dog: MasterOrderDelegate {
  func toEat(_ food: String) {
    print("\(type(of: self)) is eating \(food)")
  }
}

class Cat {}
extension Cat: MasterOrderDelegate {
  func toEat(_ food: String) {
    print("\(type(of: self)) is eating \(food)")
  }
}

let cat = Cat()
let dog = Dog()
let cat1 = Cat()

let master = Master()
// master的delegate是弱引用,所以不能直接賦值
let delegate = masterOrderDelegateManager([cat, dog])
// 添加遵守該協(xié)議的類
delegate.add(cat1)
// 刪除遵守該協(xié)議的類
delegate.remove(dog)

master.delegate = delegate
master.orderToEat()

// 輸出
// Cat is eating meat
// Cat is eating meat

設置masterOrderDelegateManager的好處是,可以通過一個數(shù)組來管理多重代理.

更多iOS相關知識點歡迎關注我的Github: SwiftTips (本地下載

總結

以上就是這篇文章的全部內(nèi)容了,希望本文的內(nèi)容對大家的學習或者工作具有一定的參考學習價值,如果有疑問大家可以留言交流,謝謝大家對腳本之家的支持。

相關文章

  • iOS把圖片緩存到本地的幾種方法(總結)

    iOS把圖片緩存到本地的幾種方法(總結)

    下面小編就為大家分享一篇iOS把圖片緩存到本地的幾種方法總結,具有很的參考價值,希望對大家有所幫助。一起跟隨小編過來看看吧
    2017-12-12
  • 詳解Swift 利用Opration和OprationQueue來下載網(wǎng)絡圖片

    詳解Swift 利用Opration和OprationQueue來下載網(wǎng)絡圖片

    這篇文章主要介紹了詳解Swift 利用Opration和OprationQueue來下載網(wǎng)絡圖片的相關資料,希望通過本文能幫助到大家,需要的朋友可以參考下
    2017-09-09
  • iOS仿網(wǎng)易簡單頭部滾動效果

    iOS仿網(wǎng)易簡單頭部滾動效果

    這篇文章主要為大家詳細介紹了iOS仿網(wǎng)易簡單頭部滾動效果,具有一定的參考價值,感興趣的小伙伴們可以參考一下
    2018-05-05
  • IOS中Json解析實例方法詳解(四種方法)

    IOS中Json解析實例方法詳解(四種方法)

    本文將介紹TouchJson、 SBJson 、JSONKit 和 iOS5所支持的原生的json方法,解析國家氣象局API。通過本文給大家介紹IOS中Json解析的四種方法,非常不錯,具有參考借鑒價值,感興趣的朋友一起學習吧
    2016-06-06
  • iOS開發(fā)數(shù)獨小游戲實例

    iOS開發(fā)數(shù)獨小游戲實例

    這篇文章主要為大家詳細介紹了iOS開發(fā)數(shù)獨小游戲實例,文中示例代碼介紹的非常詳細,具有一定的參考價值,感興趣的小伙伴們可以參考一下
    2021-11-11
  • iOS App中調用iPhone各種感應器的方法總結

    iOS App中調用iPhone各種感應器的方法總結

    Xcode環(huán)境中包含CoreMotion框架,能夠幫助我們調用硬件設備的加速度傳感器和陀螺儀等感應器,下面比較詳細地整理了iOS App中調用iPhone各種感應器的方法總結,需要的朋友可以參考下:
    2016-07-07
  • iOS Block解開多年以來一直的誤解

    iOS Block解開多年以來一直的誤解

    這篇文章主要給大家介紹了關于iOS Block多年以來一直的誤解如何解開的相關資料,文中通過圖文介紹的非常詳細,對大家的學習或者工作具有一定的參考學習價值,需要的朋友們下面隨著小編來一起學習學習吧
    2018-05-05
  • iOS監(jiān)聽手機鎖屏狀態(tài)

    iOS監(jiān)聽手機鎖屏狀態(tài)

    iPhone的鎖屏監(jiān)測分為兩種方式監(jiān)聽,本文給大家介紹的非常詳細,具體內(nèi)容詳情大家通過本文詳細了解下吧
    2017-05-05
  • iOS開發(fā)中的幾個手勢操作實例分享

    iOS開發(fā)中的幾個手勢操作實例分享

    這篇文章主要介紹了iOS開發(fā)中的幾個手勢操作實例分享,編寫代碼為傳統(tǒng)的Objective-C,需要的朋友可以參考下
    2015-09-09
  • iOS模仿電子書首頁實現(xiàn)書架布局樣式

    iOS模仿電子書首頁實現(xiàn)書架布局樣式

    這篇文章主要為大家詳細介紹了iOS實現(xiàn)類似電子書首頁效果樣式,實現(xiàn)書架布局樣式,感興趣的小伙伴們可以參考一下
    2016-03-03

最新評論