2

我正在尝试构建一组共享通用初始化代码的类。除了继承之外,我认为协议是要走的路。虽然协议和协议扩展对我来说适用于实例和静态方法,但我在使其与初始化程序一起工作时遇到了一些麻烦。

假设我们有这个协议:

protocol CloudServiceWrapper {

    static var isConfigured: Bool { get }

    init()

    func initialize()

}

现在假设我们要在协议扩展isConfigured中添加默认实现:init()

extension CloudServiceWrapper {

    static var isConfigured: Bool {
        get {
            return true
        }
    }

    init() {
        print("Initializing generic CloudServiceWrapper")
        self.init()
        if Self.isConfigured {
            initialize()
        }

    }

}

最后,让我们有一个类实现这个协议并尝试从它的默认实现中受益:

class OneDriveWrapper: CloudServiceWrapper {

    required init() {
        // CloudServiceWrapper() // error: protocol type 'CloudServiceWrapper' cannot be instantiated
        // self = CloudServiceWrapper.init() // error: cannot assign to value: 'self' is immutable
        // super.init() // error: 'super' members cannot be referenced in a root class
        // (self as CloudServiceWrapper).init() // error: static member 'init' cannot be used on instance of type 'CloudServiceWrapper'
        print("Initializing OneDriveWrapper")
    }

    func initialize() {
        print("Done")
    }

}

在尝试构建OneDriveWrapper类的新实例时,我根本找不到既调用类的初始化程序又调用默认协议实现的方法。并且不可能init()OneDriveWrapper类中省略 ,因为它是协议定义所必需的,并且似乎不被认为是通过协议扩展“实现”的。

事实上,更广泛地说,我找不到任何方法来显式调用协议扩展的初始化程序,即使我知道实例方法是可能的。

我究竟做错了什么?您知道将类的初始化程序和协议扩展的初始化程序结合起来的任何方法吗?我应该回到类继承而不是协议和扩展吗?

谢谢!

4

3 回答 3

4

a内init的 aprotocolrequired,因此必须显式实现,即不能使用默认实现。

至于“显式调用协议扩展的初始化程序”,您无法实例化协议类型。

我建议为此使用继承。

于 2018-08-13T22:00:38.673 回答
3

协议中的公共默认初始化是一个类似的问题。

尽管协议提供了默认实现,为什么一个类需要一个 init?答案是它没有提供默认的实现,而只是另一个 init,它需要init()这里的类存在。

由于您的协议不能确定覆盖使用它的类的所有成员,因此您在协议中声明的任何初始化程序都需要将类的“未知”成员的初始化委托给该类本身提供的另一个初始化程序。 资源

但是,您的协议 init 可能会被类遮蔽,init()因此您将无法使用它。也许将参数添加到类初始化器/协议所需的初始化器或参数到协议扩展ìnit()。如果协议扩展 init 没有被类隐藏,您将能够使用它来初始化类。

init 是必需的,因为该类可以是子类,并且初始化程序不一定是继承的,请参阅Initializer Inheritance and Overriding 下的继承文档和初始化文档。如果不需要 init,则子类将不再遵守协议,这是不可取的,因此在协议中设置的 init 仅由所需的初始化程序填充。

注意:在required init()您尝试调用另一个初始化程序时,但我认为除了初始化父级之外这是不可能的,因为required init()指定的初始化程序不能链接在同一个类中,只有便利初始化程序可以链接在同一个类中。

您可以将协议扩展ìnit视为便利初始化程序,因为 init 需要调用另一个初始化程序,就像所有其他便利初始化程序对self.init(...).

于 2018-08-13T22:12:08.103 回答
0

这个答案没有指出 OP 的核心问题。留在这里只是为了记录


你做错了。协议不是超类。

协议的默认实现只是在您不实现协议所需方法时使用的默认值。您不能以任何方式直接调用协议的默认实现。

协议不是超类的简单替换。根据协议改变你的思维方式。

即使我知道实例方法是可能的。

你误会了什么。即使它是一种方法,也不能调用协议的默认实现。

于 2018-08-13T21:48:36.927 回答