一尘不染

快速覆盖实例变量

swift

我知道这个答案已经以其他形式发布在这里,但是我想了解更多有关迅速覆盖实例变量的信息。

假设我有这段代码

class BaseView:UIView{
 let someVariable:Int = 1
 // do some work with someVariable
}

class ExtendedView:BaseView{
 let someVariable:Int = 2
}

好。从我读到的内容来看,常量需要覆盖前缀。其他答案说我应该声明setter和getter?为什么?我真的不在乎这两个。我只需要替换值即可。我真的不能使用init重写,因为我是从UIView继承的,这可能很危险(我认为)。

任何建议都欢迎。


阅读 233

收藏
2020-07-07

共1个答案

一尘不染

如您所说,您不能简单地在子类中重新定义常量(毕竟,它是一个常量)。您收到的错误是“无法使用存储的属性覆盖”。它似乎是可以覆盖一个var,但是,当我改变了let someVariablevar someVariable我碰到一个“暧昧使用‘someVariable’的”当我访问它的子类(注意-
同样的事情发生了,我是否使用override与否)。

最简单的解决方案是使用吸气剂。这实际上是一个函数,因此您可以愉快地重写它,将为您管理后备变量,如果您不提供设置器,则该变量对于每个类都是恒定的:

class BaseView: UIView {
    var someVariable: Int { get { return 1 } }
    // do some work with someVariable
}

class ExtendedView: BaseView {
    override var someVariable: Int { get { return 2 } }
}

let a = BaseView()
a.someVariable // 1
let b = ExtendedView()
b.someVariable // 2

正如评论员@ user3633673指出的那样,如果 只有
一个getter(而不是setter),则可以删除get,但是为了清楚起见,我将其保留了下来。没有它就一样…

class BaseView: UIView {
    var someVariable: Int { return 1 }
    // do some work with someVariable
}

class ExtendedView: BaseView {
    override var someVariable: Int { return 2 }
}

let a = BaseView()
a.someVariable // 1
let b = ExtendedView()
b.someVariable // 2

…,当然,在Swift 5中,您可以删除return

class BaseView: UIView {
    var someVariable: Int { 1 }
}

class ExtendedView: BaseView {
    override var someVariable: Int { 2 }
}
2020-07-07