Swift 拥有类似 C# 的属性声明语法:
var foo : int {
get { return getFoo() }
set { setFoo(newValue) }
}
但是,Swift 还另有 willSet
和 didSet
行为。它们分别在调用 setter 之前和之后被调用。那它们存在的意义是什么?毕竟考虑到在 setter 里可以写出完全相同的代码。
其实关键在于,有时候我们需要存储属性时自动进行某些行为,例如通知其它对象,这个属性被改变了。如果只有 get/set
,就需要另一个字段来保存改动之前的值。借助 willSet
和 didSet
,值被修改时进行的行为就不需要额外的字段了。例如,在这个例子中:
class Foo {
var myProperty: Int = 0 {
didSet {
print("myProperty 的值从 \(oldValue) 更改为 \(myProperty)")
}
}
}
每当 myProperty 被修改时都会输出旧的值和新的值。但如果只有 getter 和 setter,就需要这样做:
class Foo {
var myPropertyValue: Int = 0
var myProperty: Int {
get { return myPropertyValue }
set {
print("myProperty 的值从 \(myPropertyValue) 更改为 \(myProperty)")
myPropertyValue = newValue
}
}
}
所以 willSet
和 didSet
可以节省几行代码,并且精简了字段列表。
另外注意:根据苹果的注释,在初始化方法中设置属性不会调用 willSet 和 didSet:willSet and didSet observers are not called when a property is first initialized. They are only called when the property’s value is set outside of an initialization context.