分析iOS中的多继承与多重代理

发布时间:2021-11-04 17:38:09 作者:iii
来源:亿速云 阅读:190

这篇文章主要介绍“分析iOS中的多继承与多重代理”,在日常操作中,相信很多人在分析iOS中的多继承与多重代理问题上存在疑惑,小编查阅了各式资料,整理出简单好用的操作方法,希望对大家解答”分析iOS中的多继承与多重代理”的疑惑有所帮助!接下来,请跟着小编一起来学习吧!

1. 多继承

1. 实现过程

swift中的类可以遵守多个协议,但是只可以继承一个类,而值类型(结构体和枚举)只能遵守单个或多个协议,不能做继承操作.

多继承的实现:协议的方法可以在该协议的extension中实现

protocol Behavior { func run()}extension Behavior { func run() {  print("Running...") }}struct Dog: Behavior {}let myDog = Dog()myDog.run() // Running...

无论是结构体还是类还是枚举都可以遵守多个协议,所以要实现多继承,无非就是多遵守几个协议的问题.

下面举个例子.

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协议就可以啦!

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都有一个默认的实现方法method(),由于编译器不知道继承过来的method()方法是哪个,就会报错.

?钻石问题Diamond Problem,当某一个类或值类型在继承图谱中有多条路径时就会发生.

解决方法:

1. 在目标值类型或类中重写那个发生冲突的方法method().

2. 直接修改协议中重复的方法.

文章开头我们提到的问题2,我们可以试着用多重代理去解决这个问题.

2. 多重代理

1. 多重代理的实现过程

我们以一个代理的经典问题来表述:

主人叫宠物们去吃饭,吃这个动作作为一个协议,我们要做到统一管理.

1. 定义协议

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

2. 定义一个类: 用来管理遵守协议的类

这边用了NSHashTable来存储遵守协议的类,NSHashTable和NSSet类似,但又有所不同,总的来说有这几个特点:

1. NSHashTable中的元素可以通过Hashable协议来判断是否相等.

2. NSHashTable中的元素如果是弱引用,对象销毁后会被移除,可以避免循环引用.

class masterOrderDelegateManager : MasterOrderDelegate {  private let multiDelegate: NSHashTable<AnyObject> = NSHashTable.weakObjects()  init(_ delegates: [MasterOrderDelegate]) {    delegates.forEach(multiDelegate.add)  }  // 协议中的方法,可以有多个  func toEat(_ food: String) {    invoke { $0.toEat(food) }  }  // 添加遵守协议的类  func add(_ delegate: MasterOrderDelegate) {    multiDelegate.add(delegate)  }  // 删除指定遵守协议的类  func remove(_ delegateToRemove: MasterOrderDelegate) {    invoke {      if $0 === delegateToRemove as AnyObject {        multiDelegate.remove($0)      }    }  }  // 删除所有遵守协议的类  func removeAll() {    multiDelegate.removeAllObjects()  }  // 遍历所有遵守协议的类  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])// 添加遵守该协议的类delegate.add(cat1)// 删除遵守该协议的类delegate.remove(dog)master.delegate = delegatemaster.orderToEat()// 输出// Cat is eating meat// Cat is eating meat

设置masterOrderDelegateManager的好处是,可以通过一个数组来管理多重代理.

更多iOS相关知识点欢迎关注我的Github: SwiftTips(本地下载)

到此,关于“分析iOS中的多继承与多重代理”的学习就结束了,希望能够解决大家的疑惑。理论与实践的搭配能更好的帮助大家学习,快去试试吧!若想继续学习更多相关知识,请继续关注亿速云网站,小编会继续努力为大家带来更多实用的文章!

推荐阅读:
  1. iOS 代理与通知,kvc和kvo的区别
  2. python中怎么实现继承与多重继承

免责声明:本站发布的内容(图片、视频和文字)以原创、转载和分享为主,文章观点不代表本网站立场,如果涉及侵权请联系站长邮箱:is@yisu.com进行举报,并提供相关证据,一经查实,将立刻删除涉嫌侵权内容。

ios

上一篇:python正则表达式参数替换实例分析

下一篇:全外连接的union all改写方法是什么样的

相关阅读

您好,登录后才能下订单哦!

密码登录
登录注册
其他方式登录
点击 登录注册 即表示同意《亿速云用户服务条款》