【问题标题】:Inherit from class with private initializer?从具有私有初始化程序的类继承?
【发布时间】:2015-09-30 14:54:19
【问题描述】:

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

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

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

鉴于 Swift 中初始化程序继承的规则,我能够阻止构建 MapItem 类的唯一方法是声明一个具有不同签名的新私有初始化程序,这样基类初始化程序就不会被继承。

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

有什么方法可以快速完成我想要的。我想尽可能保持继承层次结构,因为 MapItem 包含 ClusterItem 和 Cluster 的一些通用实现代码。此外,我希望能够使用 MapItems 集合来引用 ClusterItems 和 Clusters。

【问题讨论】:

  • 设为internal,而不是private。或者在同一个文件中声明类,那么private就没有问题了。
  • 感谢@Sulthan,在同一个文件中声明类有效。来自 C++/C# 世界的 Swift 访问控制需要一段时间才能适应。

标签: swift inheritance


【解决方案1】:

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

  1. 将初始化程序声明为internal,这将使您能够访问模块并且您的子类将能够调用它。

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

  3. 除了抽象类,您可以使MapItemprotocolClusterItemCluster 可以直接从GMSMarker 继承。但是,根据您的具体需求,此解决方案可能并不理想。

【讨论】:

  • fileprivate 在现代 Swift 中。
猜你喜欢
  • 1970-01-01
  • 1970-01-01
  • 1970-01-01
  • 1970-01-01
  • 1970-01-01
  • 1970-01-01
  • 1970-01-01
  • 2013-12-13
  • 1970-01-01
相关资源
最近更新 更多