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

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

最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
平台声明:文章内容(如有图片或视频亦包括在内)由作者上传并发布,文章内容仅代表作者本人观点,简书系信息发布平台,仅提供信息存储服务。

推荐阅读更多精彩内容

  • 发现 关注 消息 iOS 第三方库、插件、知名博客总结 作者大灰狼的小绵羊哥哥关注 2017.06.26 09:4...
    肇东周阅读 12,267评论 4 61
  • 转至元数据结尾创建: 董潇伟,最新修改于: 十二月 23, 2016 转至元数据起始第一章:isa和Class一....
    40c0490e5268阅读 1,837评论 0 9
  • 前言 对于在饭圈遇到的一点事情的一点看法。不准,勿撕。 之前的成都接机事件给我感觉就是满城风雨尽袭来,但是我也不想...
    萤窗阅读 305评论 0 1
  • 一 我身边有太多走散了的情侣。 他们有的乍见之欢久处不厌,有的互相扶持几年道了句再见。 不管是哪一种,都一次一次更...
    阿嚏鱼阅读 541评论 6 6
  • 安装mysql sudo yum install mysql mysql-server mysql-devel 安...
    cws阅读 2,748评论 1 3