swift 中的KVO用法

本文介绍了Swift中的Key-Value Observing (KVO)机制,它基于Objective-C Runtime,用于在对象属性变化时通知观察者。同时,文章讨论了属性观察器,包括willSet和didSet,它们在属性值改变时自动触发,允许在设置新值前后进行操作。KVO和属性观察器在某些场景下是监测和响应属性变化的有效工具。

摘要生成于 C知道 ,由 DeepSeek-R1 满血版支持, 前往体验 >

KVO 

KVO (Key-Value Observing)

KVO 是 Objective-C 对观察者模式(Observer Pattern)的实现。也是 Cocoa Binding 的基础。当被观察对象的某个属性发生更改时,观察者对象会获得通知。

KVO 实现机制

KVO 的实现也依赖于 Objective-C 强大的 Runtime 。

例:监听 button 的 backGroundColor 属性

import UIKit

class ViewController: UIViewController {

    @IBOutlet var btn: UIButton!

     var myContext:NSObject!

    override func viewDidLoad() {
        super.viewDidLoad()
        myContext = NSObject()

        btn.addObserver(self, forKeyPath: "backgroundColor", options: .new, context: &myContext)
    }

    @IBAction func btnClicked(_ sender: Any) {
        btn.backgroundColor = UIColor.red
    }

    override func observeValue(forKeyPath keyPath: String?, of object: Any?, change: [NSKeyValueChangeKey : Any]?, context: UnsafeMutableRawPointer?) {
        if context == &myContext {
            if let newValue = change?[NSKeyValueChangeKey.newKey] {
                print("Date changed: \(newValue)")
            }
        } else {
            super.observeValue(forKeyPath: keyPath, of: object, change: change, context: context)
        }
    }

    deinit {
        btn.removeObserver(self, forKeyPath: "titleColor", context: &myContext)
    }

}


点击button,打印结果: 

 Date changed: UIExtendedSRGBColorSpace 1 0 0 1

属性观察器

1、简单介绍

属性观察器相当于內建的KVO,监控和响应属性值的变化,每次属性被设置值的时候都会调用属性观察器,甚至新的值和现在的值相同的时候也不例外。

可以为除了延迟存储属性之外的其他存储属性添加属性观察器,也可以通过重载属性的方式为继承的属性(包括存储属性和计算属性)添加属性观察器。

注意:

不需要为无法重载的计算属性添加属性观察器,因为可以通过 setter 直接监控和响应值的变化。 

2、监测方式

可以为属性添加如下的一个或全部观察器:

  • willSet在设置新的值之前调用
  • didSet在新的值被设置之后立即调用

分析:

willset观察器会将新的属性值作为固定参数传入,在willSet的实现代码中可以为这个参数指定一个名称,如果不指定则参数仍然可用,这时使用默认名称

newValue表示。类似地,didSet观察器会将旧的属性值作为参数传入,可以为该参数命名或者使用默认参数名oldValue。

注意:

willSet和didSet观察器在属性初始化过程中不会被调用,它们只会当属性的值在初始化之外的地方被设置时被调用。

写法:

class PropertyObserverExmple {
    var number: Int = 0 {
        willSet(newNumber) {
            print("About to change to \(newNumber)")
        }
        didSet(oldNumber) {
            print("Just changed from \(oldNumber) to \(self.number)")
        }
    }
}

var observer = PropertyObserverExmple()
observer.number = 4
打印结果:

About to change to 4

Just changed from 0 to 4


使用场景举例:

更换UI主题模式




评论 2
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值