【问题标题】:Default implementation of protocol implementing protocol协议实现协议的默认实现
【发布时间】:2016-02-09 12:26:49
【问题描述】:

我正在尝试创建一个由某些类实现的协议,它们都应该实现UIScrollViewDelegate。我想到的是让我的新协议实现协议UIScrollViewDelegate

protocol MyProtocol: UIScrollViewDelegate {
    var myVar: NSString { get }
    func myMethod()
}

由于协议应该有它的默认实现,我还为此协议创建了一个扩展。

extension MyProtocol {
    func myMethod() {
        print("I'm printing")
    }

    func scrollViewDidScroll(scrollView: UIScrollView) {
        print("I'm scrollin")
    }
}

它可以编译,但不起作用。我做错了什么以及创建扩展协议的默认实现的正确方法是什么?

【问题讨论】:

  • 我认为你是在正确的轨道上。我的猜测是您必须在协议中覆盖委托的设置器并调用 super.delegate = theDelegate
  • “它不起作用”是什么意思。发生什么了?这不是对问题的可接受描述。
  • 乍一看,您走在了正确的轨道上,但从 Swift 2.2 开始,在协议扩展中定义的委托方法不会从 Objective-C 代码中“找到”。需要扩展符合UIScrollViewDelegate的类,即在具体类MyScrollViewDelegateMyScrollViewViewController中定义扩展方法scrollViewDidScroll
  • @nhgrif 表示没有进入函数

标签: ios swift uiscrollviewdelegate swift-extensions swift-protocols


【解决方案1】:

你想要做的是:

protocol MyProtocol{
    var myVar: NSString { get }
    func myMethod()
}

protocol MyProtocol2{
    var myVar2: NSString { get }
    func myMethod2()
}

extension MyProtocol where Self: MyProtocol2 {
    func myMethod() {
        print("I'm printing ")
    }
}

class anotherClass: MyProtocol, MyProtocol2 {
    var myVar: NSString {
        return "Yo"
    }

    var myVar2: NSString {
        return "Yo2"
    }

    func myMethod2() {
        print("I'm printing in myMethod2")
    }
}

在上面的代码中MyProtocol2相当于你的UIScrollViewDelegate,

因此你要做的是:

protocol MyProtocol{
    var myVar: NSString { get }
    func myMethod()
}

extension MyProtocol where Self: UIScrollViewDelegate {
    func myMethod() {
        print("I'm printing")
    }
}

class anotherClass: NSObject, MyProtocol, UIScrollViewDelegate {
    var myVar: NSString {
        return "Yo"
    }
}

注意另一个类继承了NSObject,这是因为如果你不这样做,你会得到错误

anotherClass 不符合协议NSObjectProtocol

这个错误是因为UIScrollViewDelegate本身被定义为扩展NSObjectProtocol,这是一个由NSObject实现的objective-C协议。

所以让你的类继承自 NSObject 以符合 NSObjectProtocol。 Vanilla Swift 类没有。

【讨论】:

  • 我想将此函数实现移出类的主体,因为它应该在每个实现协议的类中以相同的方式运行。我知道这不是协议的用途,但我想创建一个协议扩展,并实现其他协议(由主要协议实现)的方法。
  • 是的,你仍然可以在协议扩展中实现 UIScrollViewDelegate 的方法
  • 感谢@uchiha 的回答我尝试这样做是因为anotherClassUIViewController 的子类,它已经是NSObject 的后代,但是@ 中的scrollViewDidScroll(_ scrollView: UIScrollView) 函数987654331@ 未被正确覆盖。好像我只在我的课堂上添加了另一个版本的scrollViewDidScroll。我知道通过您描述的方法,您将能够满足一个协议与另一个协议的功能,但是您是否能够为 scrollViewDidScroll 函数创建默认实现?
最近更新 更多