addObserver形式
创建观察者并添加观察
class People: NSObject {
@objc dynamic var name:String?
@objc dynamic var sex:String?
init(name:String, sex:String = "M") {
self.name = name
self.sex = sex
super.init()
for keyPath in ["name", "sex"] {
addObserver(self, forKeyPath: keyPath, options: [.new, .old], context: nil)
}
}
override func observeValue(forKeyPath keyPath: String?, of object: Any?, change: [NSKeyValueChangeKey : Any]?, context: UnsafeMutableRawPointer?) {
debugPrint("keyPath = \(keyPath!) old = \(change![NSKeyValueChangeKey.oldKey]!) new = \(change![NSKeyValueChangeKey.newKey]!)")
}
deinit {
for keyPath in ["name", "sex"] {
removeObserver(self, forKeyPath: keyPath)
}
}
}
调用
var pp: People?
pp = People(name: "飞", sex: "W")
pp?.name = "凤"
pp?.sex = "👱"
输出结果
"keyPath = name old = 飞 new = 凤"
"keyPath = sex old = W new = 👱"
observe 形式
添加
class People: NSObject {
@objc dynamic var name:String?
@objc dynamic var sex:String?
var nameKeyValueObservation:NSKeyValueObservation?
var sexKeyValueObservation:NSKeyValueObservation?
init(name:String, sex:String = "M") {
self.name = name
self.sex = sex
super.init()
nameKeyValueObservation = observe(\People.name, options: [.new, .old], changeHandler: { (self, change) in
debugPrint("newValue = \(String(describing: change.newValue)) oldValue = \(String(describing: change.oldValue))")
})
let _ = observe(\People.name, options: [.new, .old], changeHandler: { (self, change) in
debugPrint("不起作用")
debugPrint("newValue = \(String(describing: change.newValue)) oldValue = \(String(describing: change.oldValue))")
})
sexKeyValueObservation = observe(\.sex, options: [.new, .old], changeHandler: { (self, change) in
debugPrint("newValue = \(String(describing: change.newValue)) oldValue = \(String(describing: change.oldValue))")
})
}
}
调用
var pp: People?
pp = People(name: "飞", sex: "W")
pp?.name = "凤"
pp?.sex = "👱"
输出
"newValue = Optional(Optional(\"凤\")) oldValue = Optional(Optional(\"飞\"))"
"newValue = Optional(Optional(\"👱\")) oldValue = Optional(Optional(\"W\"))"
属性观察器
添加
class People: NSObject {
private var _name:String = ""
private var _sex:String = ""
var name:String = "" {
willSet {
debugPrint("newVaule = \(newValue)")
_name = newValue
}
didSet {
debugPrint("oldValue = \(oldValue)")
}
}
var sex:String = "" {
willSet {
debugPrint("newVaule = \(newValue)")
_sex = newValue
}
didSet {
debugPrint("oldValue = \(oldValue)")
}
}
}
调用
pp = People()
pp?.name = "🐉"
pp?.sex = "W"
输出
"newVaule = 🐉"
"oldValue = "
"newVaule = W"
"oldValue = "
Tips
- Swift值Key-Path表达式:
Key-Path表达式通常用来引用一个类型的属性和下标,可以在KVO等动态编程中使用 。基本形式为“\type name .path”; 类型名 是一个具体类型的名称包含泛型类型,路径可以右属性 ,下标,可选链式表达式和强制解包表达式构成。在编译期Key-Path表达式会被 keyPath类的实例代替
- OC是基于运行时的,遵循了KVC和Dynamic Dispatch。Swift为了性能考虑使用的是静态派发。
- Swift使用 dynamic修饰告诉编译期使用动态派发
【推荐】国内首个AI IDE,深度理解中文开发场景,立即下载体验Trae
【推荐】编程新体验,更懂你的AI,立即体验豆包MarsCode编程助手
【推荐】抖音旗下AI助手豆包,你的智能百科全书,全免费不限次数
【推荐】轻量又高性能的 SSH 工具 IShell:AI 加持,快人一步
· 开发者必知的日志记录最佳实践
· SQL Server 2025 AI相关能力初探
· Linux系列:如何用 C#调用 C方法造成内存泄露
· AI与.NET技术实操系列(二):开始使用ML.NET
· 记一次.NET内存居高不下排查解决与启示
· Manus重磅发布:全球首款通用AI代理技术深度解析与实战指南
· 被坑几百块钱后,我竟然真的恢复了删除的微信聊天记录!
· 没有Manus邀请码?试试免邀请码的MGX或者开源的OpenManus吧
· 园子的第一款AI主题卫衣上架——"HELLO! HOW CAN I ASSIST YOU TODAY
· 【自荐】一款简洁、开源的在线白板工具 Drawnix