背景
最近遇到了一个需求,一个页面上好几个按钮都依赖于某一个model的变化而做相应的变化,每次在model变化的时候分别对各个按钮做更新倒是也可以,就是有点麻烦,于是我想到了KVO,当model变化的时候。自动通知需要知道的按钮,让按钮自身去做这件事。然而,事情并不那么简单。
基础
keyPath
首先了解下keyPath,因为可能很多同学都卡在这里了。在 Swift 中,keyPath是用来表达访问属性时的关键路径。什么是路径呢?假设我们有一个Person类和一个Person类的实例,例如:
class Person {
var sex = 0
}
let someone = Person()
我们访问实例的属性时会用someone.sex 的语句, 这里的someone.sex 就是表达式访问人物类的性别属性的路径。
那么keyPath怎么表达这个路径呢,有三种方式:
- 使用键路径表达式 \Person.sex
这种方式里,我们希望访问someone这个实例的sex属性。应该先语法,然后写实例的类和.运算符,最后写我们希望观察的属性。 - 使用 #keyPath(Person.sex))
这里的#keyPath 表达式的语法很像选择器表达式(#selector)的语法,其中使用这种方法获取对象的属性值时,属性要 @objc 修饰,否则会报错。 - 字符串
字符串的方式可以直接写要观察的属性路径,不需要写根类型,例如观察sex属性的话可以直接写"sex"
KVO的前提
- KVO是NSObject才有的功能,也就是说,如果像我们上边写的Person类,它没有继承NSObject的话,是不支持KVO的。
- KVO的属性需要是@objc, 'dynamic' 修饰的, 如果是像我们上边提到的sex属性,在编译的时候就会提示我们如下错误
Cannot convert value of type 'ReferenceWritableKeyPath<Person, Int>' to expected argument type 'String'
实践
假设我们有一个对象想要观察Person实例的sex属性的话。应该怎么操作呢
- 首先我们为实例添加一个观察者
let someone = Person()
someone.sex = 0
someone.addObserver(self, forKeyPath: \Person.sex, options: [.new], context: nil)
- 然后我们重写observe函数 这里简单的把keyPath打印了出来
override class func observeValue(forKeyPath keyPath: String?, of object: Any?, change: [NSKeyValueChangeKey : Any]?, context: UnsafeMutableRawPointer?) {
print(keyPath)
}
3.然后我们试着改变一下sex的值
someone.sex = 1
因为我们观察的时候选择了观察新值([.new])所以。我们会在observeValue(forKeyPath keyPath: String?, of object: Any?, change: [NSKeyValueChangeKey : Any]?, context: UnsafeMutableRawPointer?) 里收到sex的新值1,如果大家没有收到的话。检查一下属性是否正确使用了修饰符。
4.释放
一定要在不需要观察的时候释放这个观察者
deinit {
some.removeObserver(self)
}
总结
观察者使用起来API非常简单,但是一定要注意防止造成循环引用。