使用私有初始化程序从类继承?

时间:2015-09-30 14:54:19

标签: swift inheritance

考虑Swift中的以下类层次结构:

Class Hierarchy

GMSMarker是GoogleMaps库提供的课程。它有两个公共初始化器(一个指定,一个便利)。

MapItem,ClusterItem和Cluster是我模型的一部分。我不想允许构建MapItem对象;只有ClusterItems和Clusters。由于Swift没有抽象类,因此拥有私有初始化程序就足够了。但是,鉴于MapItem使用便利构造函数从GMSMarker继承,我不能简单地将指定的初始化程序覆盖为私有。

鉴于Swift中初始化程序继承的规则,我能够阻止构造MapItem类的唯一方法是声明一个具有不同签名的新私有初始化程序,以便基类初始化程序不是&#39 ; t继承。

class MapItem : GMSMarker {

    private init(dummyParam: Int) {
        super.init()
    }
}

到目前为止一切顺利。现在,在尝试为ClusterItem类创建初始化程序时,我遇到了一个问题。

class ClusterItem : MapItem {

    weak var post: Post?

    init(post: Post) {
        self.post = post
        super.init(dummyParam: 0)
    }
}

我在调用super.init()的行上收到以下编译器错误:'MapItem' does not have a member named 'init'

如果我省略对super.init的调用,则会收到以下错误:Super.init isn't called before returning from initializer

有没有办法在swift中完成我想要的东西。我希望尽可能保留继承层次结构,因为MapItem包含ClusterItem和Cluster的一些常见实现代码。此外,我希望能够使用MapItems集合引用ClusterItems和Clusters。

1 个答案:

答案 0 :(得分:1)

有多种方法可以解决这个问题:

  1. 将初始化程序声明为internal,它将为您提供模块访问权限,您的子类将能够调用它。

  2. 在同一个文件中声明这些类,然后private不再是问题(private允许从同一个文件访问)。

  3. 您可以使MapItem成为protocolClusterItemCluster可以直接从GMSMarker继承,而不是抽象类。但是,根据您的具体需求,此解决方案可能不太好。