溫馨提示×

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

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

怎么在iOS中實(shí)現(xiàn)多繼承與多重代理

發(fā)布時(shí)間:2021-06-11 16:59:24 來源:億速云 閱讀:334 作者:Leah 欄目:移動(dòng)開發(fā)

本篇文章給大家分享的是有關(guān)怎么在iOS中實(shí)現(xiàn)多繼承與多重代理,小編覺得挺實(shí)用的,因此分享給大家學(xué)習(xí),希望大家閱讀完這篇文章后可以有所收獲,話不多說,跟著小編一起來看看吧。

1. 多繼承

1. 實(shí)現(xiàn)過程

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

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

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

struct Dog: Behavior {}

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

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

下面舉個(gè)例子.

2. 通過多繼承為UIView擴(kuò)展方法

// 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()

怎么在iOS中實(shí)現(xiàn)多繼承與多重代理

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

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

請(qǐng)看下面代碼

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 {}

此時(shí)ProtocolA和ProtocolB都有一個(gè)默認(rèn)的實(shí)現(xiàn)方法method(),由于編譯器不知道繼承過來的method()方法是哪個(gè),就會(huì)報(bào)錯(cuò).

?鉆石問題Diamond Problem,當(dāng)某一個(gè)類或值類型在繼承圖譜中有多條路徑時(shí)就會(huì)發(fā)生.

解決方法:

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

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

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

2. 多重代理

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

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

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

1. 定義協(xié)議

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

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

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

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

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

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

  init(_ delegates: [MasterOrderDelegate]) {
    delegates.forEach(multiDelegate.add)
  }
  
  // 協(xié)議中的方法,可以有多個(gè)
  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

以上就是怎么在iOS中實(shí)現(xiàn)多繼承與多重代理,小編相信有部分知識(shí)點(diǎn)可能是我們?nèi)粘9ぷ鲿?huì)見到或用到的。希望你能通過這篇文章學(xué)到更多知識(shí)。更多詳情敬請(qǐng)關(guān)注億速云行業(yè)資訊頻道。

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

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

ios
AI