前言
在写swift版的上下拉刷新,语法看了3个小时就开始动手了,结果一路坑,先写几篇铺垫的文章,后续文章会附上上下拉刷新有关代码。
KVO
Swift对KVO支持可谓差到不行,只能NSObject的子类进行监听,而对基本类型则不执行,写的时候踩了坑,还不知道为什么,尴尬。下面记录踩过的坑,提醒后来者。
起因是这样,我在画一个贝塞尔曲线,想监听控制点的y坐标,结果我写出下面代码(下面例子没有写removeObserver方法,你们一定要记得写哈):
public class SpringView: UIView {
var pointY = 0.0
let keyY = "pointY"
override public init(frame: CGRect) {
super.init(frame: frame)
addObserver(self, forKeyPath: keyY, options: [.old, .new], context: nil)
pointY = 1.0
}
override public func observeValue(forKeyPath keyPath: String?, of object: Any?, change: [NSKeyValueChangeKey : Any]?, context: UnsafeMutableRawPointer?) {
if keyPath == keyY {
print(pointY)
}
}
required public init?(coder aDecoder: NSCoder) {
fatalError("init(coder:) has not been implemented")
}
}
呀?!为毛不走!就是因为pointY是基本类型,此处耽误一个小时。改一改:
public class SpringView: UIView {
dynamic var point = UIView()
let keyY = "point.frame"
override public init(frame: CGRect) {
super.init(frame: frame)
addObserver(self, forKeyPath: keyY, options: [.old, .new], context: nil)
point.frame = CGRect(x: 10.0, y: 10.0, width: 3.0, height: 3.0)
}
override public func observeValue(forKeyPath keyPath: String?, of object: Any?, change: [NSKeyValueChangeKey : Any]?, context: UnsafeMutableRawPointer?) {
if keyPath == keyY {
print(point.frame)
}
}
required public init?(coder aDecoder: NSCoder) {
fatalError("init(coder:) has not been implemented")
}
}
这次监听了一个view的frame,妥。值得注意的一点,在需要KVO的属性前面要加上** dynamic**前缀,告诉编译器这个是可能被动态调用的。上述代码去掉dynamic一样正常执行,这又是为什么?
这就得说说OC和Swift的不同之处了,OC是动态语言,我们所谓的执行方法, 实际上是发消息,而Swift可以静态调用方法。当继承 NSObject 时,就认为是 Objective-C 的类,就按照 Objective-C 的方式处理。注意就算是继承 NSObject,但将方法标记成私有 private,也是不能动态找到方法地址的。所以,上述代码中,如果point是private的,不加dynamic就会报错。
那么如何监听基本类型呢?各位自己想办法,啊哈哈,给一篇文章供大家参考KVO。
上述代码中,我要监听的都是本类中的属性,就真的拿基本类型的属性没招了么?那不行,于是我有尝试了下述方法,还是回到最初情况:
public class SpringView: UIView {
var pointY = 0.0 {
didSet {
print(pointY)
}
}
override public init(frame: CGRect) {
super.init(frame: frame)
changePointY()
}
func changePointY() {
pointY = 1.0
}
required public init?(coder aDecoder: NSCoder) {
fatalError("init(coder:) has not been implemented")
}
}
Swift为我们提供了willSet和didSet两种监听器,看名字都知道这是干嘛用的,就不细说了哈。注意一点,在构造器中直接为属性赋值不会走听器的,可以使用上图代码方法或者在监听器中使用KVC赋值。
以上是我踩坑的经过,希望能给大家一些帮助。