Swift KVO相关

前言

在写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赋值。

以上是我踩坑的经过,希望能给大家一些帮助。

你可能感兴趣的:(Swift KVO相关)