KVO (Key-Value Observing) 是 Cocoa 中公认的最强大的特性之一,但是同时它也以烂到家的 API 和极其难用著称。和属性观察不同,KVO 的目的并不是为当前类的属性提供一个钩子方法,而是为了其他不同实例对当前的某个属性 (严格来说是 keypath) 进行监听时使用的。其他实例可以充当一个订阅者的角色,当被监听的属性发生变化时,订阅者将得到通知。
在 Swift 中我们也是可以使用 KVO 的,观察者和被观察者都必须是 NSObject 的子类。这是可以理解的,因为 KVO 是基于 KVC (Key-Value Coding) 以及动态派发技术实现的,而这些东西都是 Objective-C 运行时的概念,这也意味着 Swift 中强大的 Struct,Enum以及泛型都与 KVO 无缘了。另外由于 Swift 为了效率,默认禁用了动态派发,因此想用 Swift 来实现 KVO,我们还需要做额外的工作,那就是将想要观测的对象标记为 dynamic,表示该属性的存取都由 runtime 在运行时来决定。除此之外,在 NSObject 子类中几乎没有属性默认是使用@dynamic 修饰(该关键字最常见场景是在 Core Data 里, NSManagedObject 子类的属性都是 dynamic 的),所以若想对某个属性进行观察,还必须在当前的子类中 override 该属性,override 时,采用 super 的实现即可。
所以Swift要使用KVO必须满足:
- 观察者和被观察者都必须是 NSObject 的子类
- 监听属性必须标记dynamic
原理:
- 当某个类的对象第一次被观察时,系统就会在运行期动态地创建该类的一个派生类,在这个派生类中重写基类中任何被观察属性的 setter 方法。
- 派生类在被重写的 setter 方法实现真正的通知机制,键值观察通知依赖于 NSObject 的两个方法: willChangeValueForKey: 和 didChangevlueForKey: 。在一个被观察属性发生改变之前, willChangeValueForKey: 一定会被调用,这就 会记录旧的值。而当改变发生后, didChangeValueForKey: 会被调用,继而 observeValueForKey:ofObject:change:context: 也会被调用。可以手动实现这些调用,但很少有人这么做。这么做是基于设置属性会调用 setter 方法,而通过重写就获得了 KVO 需要的通知机制。当然前提是要通过遵循 KVO 的属性设置方式来变更属性值,如果仅是直接修改属性对应的成员变量,是无法实现 KVO 的。
- 同时派生类还重写了 class 方法以“欺骗”外部调用者它就是起初的那个类。
- 系统将这个对象的 isa 指针指向这个新诞生的派生类,因此这个对象就成为该派生类的对象了,因而在该对象上对 setter 的调用就会调用重写的 setter,从而激活键值通知机制。
- 派生类还重写了 dealloc 方法来释放资源。
看下面代码:
-----------------start------------
对象变量名字: people
对象: <TestKVOKVC.People: 0x7fa449c14a60>
类: TestKVOKVC.People
元类: TestKVOKVC.People
实现的方法: setAge:, setSex:, age, sex, initWithName:age:sex:address:, name, address, .cxx_destruct, init, setName:, setAddress:,
-----------------end------------
-----------------start------------
对象变量名字: namePeople
对象: <TestKVOKVC.People: 0x7fa449c61e20>
类: TestKVOKVC.People
元类: NSKVONotifying_TestKVOKVC.People
实现的方法: setAge:, setName:, class, dealloc, _isKVOA,
-----------------end------------
-----------------start------------
对象变量名字: nameAgePeople
对象: <TestKVOKVC.People: 0x7fa449cbe990>
类: TestKVOKVC.People
元类: NSKVONotifying_TestKVOKVC.People
实现的方法: setAge:, setName:, class, dealloc, _isKVOA,
-----------------end------------
0x00000001083a3c60
0x00000001089807fa
0x00000001083a3c60
0x00000001089807fa
可以看到输出类名始终为:TestKVOKVC.People,这是因为新诞生的派生类重写了 -class 方法声称它就是起初的基类,只有使用 runtime 函数 object_getClass 才能一睹芳容:NSKVONotifying_TestKVOKVC.People。注意看:name,age 两个被观察对象真正的类型都是 NSKVONotifying_TestKVOKVC.People,而且该类实现了:setAge:, setName:, class, dealloc, _isKVOA 这些方法。其中 setAge:, setName:重写实现通知机制, class声明原来的类 , dealloc释放资源,私有方法 _isKVOA 是用来标示该类是一个 KVO 机制声称的类。在这里 swift 做了一些优化,它对所有被观察对象只生成一个派生类,该派生类实现所有被观察对象的 setter 方法,这样就减少了派生类的数量,提供了效率。所有 NSKVONotifying_TestKVOKVC.People 这个派生类重写了 setAge,setName方法(留意:没重写其他set方法)。
接着来看最后两行输出,地址 0x00000001083a3c60 是 TestKVOKVC.People 类中的实现,而地址是 0x00000001089807fa 是派生类 NSKVONotifying_TestKVOKVC.People 类中的实现。那后面那个地址到底是什么呢?可以通过 lldb 的 info 命令 image lookup --address 0x00000001089807fa 查看该地址的信息:
(lldb) image lookup --address 0x00000001089807fa
Address: Foundation[0x00000000000637fa] (Foundation.__TEXT.__text + 401690)
Summary: Foundation`_NSSetObjectValueAndNotify
看起来它是 Foundation 框架提供的私有函数:_NSSetObjectValueAndNotify。更进一步,我们来看看 Foundation 到底提供了哪些用于 KVO 的辅助函数。打开 terminal,使用 nm -a 命令查看 Foundation 中的信息:
nm -a /System/Library/Frameworks/Foundation.framework/Versions/C/Foundation
其中查找到我们关注的函数:
00000000000c76bf t __NSSetBoolValueAndNotify
0000000000152ab2 t __NSSetBoolValueForKeyInIvar
00000000000c74c8 t __NSSetBoolValueForKeyWithMethod
00000000000997db t __NSSetCharValueAndNotify
0000000000064759 t __NSSetCharValueForKeyInIvar
00000000000646f1 t __NSSetCharValueForKeyWithMethod
000000000001bd73 t __NSSetCheckSize
00000000003f7a50 b __NSSetClass
00000000001019af t __NSSetDirectory2
00000000000ab4d3 t __NSSetDoubleValueAndNotify
0000000000152b5d t __NSSetDoubleValueForKeyInIvar
00000000000abf65 t __NSSetDoubleValueForKeyWithMethod
0000000000139763 T __NSSetExceptionRaiser
0000000000180650 t __NSSetFileModificationUNIXTime
00000000000c9cca t __NSSetFloatValueAndNotify
0000000000152bb6 t __NSSetFloatValueForKeyInIvar
0000000000152712 t __NSSetFloatValueForKeyWithMethod
00000000000c9ba9 t __NSSetIntValueAndNotify
0000000000152c0f t __NSSetIntValueForKeyInIvar
00000000000abefb t __NSSetIntValueForKeyWithMethod
00000000001817a7 T __NSSetLogCStringFunction
00000000000a5389 t __NSSetLongLongValueAndNotify
0000000000152d5f t __NSSetLongLongValueForKeyInIvar
00000000000a55e1 t __NSSetLongLongValueForKeyWithMethod
000000000015abd7 t __NSSetLongValueAndNotify
0000000000152cb7 t __NSSetLongValueForKeyInIvar
0000000000152779 t __NSSetLongValueForKeyWithMethod
000000000011651f T __NSSetMainBundle
00000000001531ad t __NSSetNilValueForKey
00000000000647b0 t __NSSetObjectSetIvarValueForKeyInIvar
0000000000096e38 t __NSSetObjectValueAndNotify
0000000000028b2d t __NSSetObjectValueForKeyInIvar
00000000000c0657 t __NSSetPointValueAndNotify
0000000000152e5f t __NSSetPointValueForKeyInIvar
00000000000c05f1 t __NSSetPointValueForKeyWithMethod
000000000015b067 t __NSSetRangeValueAndNotify
0000000000152ebc t __NSSetRangeValueForKeyInIvar
00000000001528b2 t __NSSetRangeValueForKeyWithMethod
0000000000099610 t __NSSetRectValueAndNotify
0000000000152f15 t __NSSetRectValueForKeyInIvar
000000000015291d t __NSSetRectValueForKeyWithMethod
000000000015ae1f t __NSSetShortValueAndNotify
0000000000152db3 t __NSSetShortValueForKeyInIvar
0000000000152849 t __NSSetShortValueForKeyWithMethod
00000000000af692 t __NSSetSizeValueAndNotify
0000000000152f86 t __NSSetSizeValueForKeyInIvar
00000000000af62d t __NSSetSizeValueForKeyWithMethod
000000000015aab3 t __NSSetUnsignedCharValueAndNotify
0000000000152b09 t __NSSetUnsignedCharValueForKeyInIvar
00000000001526a9 t __NSSetUnsignedCharValueForKeyWithMethod
00000000000e59a2 t __NSSetUnsignedIntValueAndNotify
0000000000152c63 t __NSSetUnsignedIntValueForKeyInIvar
0000000000103f5e t __NSSetUnsignedIntValueForKeyWithMethod
0000000000096fa8 t __NSSetUnsignedLongLongValueAndNotify
00000000000f5f93 t __NSSetUnsignedLongLongValueForKeyInIvar
00000000000c745d t __NSSetUnsignedLongLongValueForKeyWithMethod
000000000015acfa t __NSSetUnsignedLongValueAndNotify
0000000000152d0b t __NSSetUnsignedLongValueForKeyInIvar
00000000001527e1 t __NSSetUnsignedLongValueForKeyWithMethod
000000000015af43 t __NSSetUnsignedShortValueAndNotify
0000000000152e09 t __NSSetUnsignedShortValueForKeyInIvar
0000000000103ef5 t __NSSetUnsignedShortValueForKeyWithMethod
Foundation 提供了大部分基础数据类型的辅助函数,此外还包括一些常见的 Cocoa 结构体如 Point, Range, Rect, Size,这表明这些结构体也可以用于自动键值观察,但要注意除此之外的结构体就不能用于自动键值观察了。
// // People.swift // TestKVOKVC // // Created by yangjun zhu on 15/9/12. // Copyright © 2015年 Cactus. All rights reserved. // import Foundation class People: NSObject{ dynamic var name: String dynamic var age: Int dynamic var sex: Int var address: Address? init(name: String, age: Int, sex: Int, address: Address){ self.name = name self.age = age self.sex = sex self.address = address } }
// // PrincipleViewController.swift // TestKVOKVC // // Created by yangjun zhu on 15/9/12. // Copyright © 2015年 Cactus. All rights reserved. // import UIKit class PrincipleViewController: UIViewController { let people = People(name: "Owen", age: 1, sex: 1, address: Address()) let namePeople = People(name: "Owen", age: 1, sex: 1, address: Address()) let nameAgePeople = People(name: "Owen", age: 1, sex: 1, address: Address()) deinit{ self.namePeople .removeObserver(self, forKeyPath: "name") self.nameAgePeople .removeObserver(self, forKeyPath: "name") self.nameAgePeople .removeObserver(self, forKeyPath: "age") } private var myContext = 0 override func viewDidLoad() { super.viewDidLoad() // Do any additional setup after loading the view, typically from a nib. self.namePeople.addObserver( self, forKeyPath: "name", options: NSKeyValueObservingOptions([.New, .Old]), context: &myContext) self.nameAgePeople.addObserver( self, forKeyPath: "name", options: NSKeyValueObservingOptions([.New, .Old]), context: &myContext) self.nameAgePeople.addObserver( self, forKeyPath: "age", options: NSKeyValueObservingOptions([.New, .Old]), context: &myContext) self.printDescription("people", obj: people) self.printDescription("namePeople",obj: namePeople) self.printDescription("nameAgePeople",obj: nameAgePeople) } override func didReceiveMemoryWarning() { super.didReceiveMemoryWarning() // Dispose of any resources that can be recreated. } private func classMethodNames(c: AnyObject) -> [String] { var arr = [String]() var methodCount: CUnsignedInt = 0; let methodList = class_copyMethodList(object_getClass(c), &methodCount); for i in 0...methodCount { arr.append( NSStringFromSelector(method_getName(methodList[Int(i)]))) } free(methodList); return arr; } private func printDescription(objectName: String, obj: AnyObject) { print("-----------------start------------") print("对象变量名字:",objectName) print("对象:",obj) print("类:",NSStringFromClass(obj.classForCoder)) print("元类:",NSStringFromClass(object_getClass(obj))) print("实现的方法:",self.classMethodNames(obj).joinWithSeparator(", ")) print("-----------------end------------") } /* func observeValueForKeyPath(keyPath: String?, object: AnyObject?, change: [String : AnyObject]?, context: UnsafeMutablePointer<Void>) { if let change = change where context == &myContext{ print(keyPath, "改变了") print(keyPath, "new:" , change[NSKeyValueChangeNewKey]) print(keyPath, "old:" , change[NSKeyValueChangeOldKey]) return; } super.observeValueForKeyPath(keyPath, ofObject: object, change: change, context: context) } */ }
输出:
-----------------start------------
对象变量名字: people
对象: <TestKVOKVC.People: 0x7fa449c14a60>
类: TestKVOKVC.People
元类: TestKVOKVC.People
实现的方法: setAge:, setSex:, age, sex, initWithName:age:sex:address:, name, address, .cxx_destruct, init, setName:, setAddress:,
-----------------end------------
-----------------start------------
对象变量名字: namePeople
对象: <TestKVOKVC.People: 0x7fa449c61e20>
类: TestKVOKVC.People
元类: NSKVONotifying_TestKVOKVC.People
实现的方法: setAge:, setName:, class, dealloc, _isKVOA,
-----------------end------------
-----------------start------------
对象变量名字: nameAgePeople
对象: <TestKVOKVC.People: 0x7fa449cbe990>
类: TestKVOKVC.People
元类: NSKVONotifying_TestKVOKVC.People
实现的方法: setAge:, setName:, class, dealloc, _isKVOA,
-----------------end------------
0x00000001083a3c60
0x00000001089807fa
0x00000001083a3c60
0x00000001089807fa
可以看到输出类名始终为:TestKVOKVC.People,这是因为新诞生的派生类重写了 -class 方法声称它就是起初的基类,只有使用 runtime 函数 object_getClass 才能一睹芳容:NSKVONotifying_TestKVOKVC.People。注意看:name,age 两个被观察对象真正的类型都是 NSKVONotifying_TestKVOKVC.People,而且该类实现了:setAge:, setName:, class, dealloc, _isKVOA 这些方法。其中 setAge:, setName:重写实现通知机制, class声明原来的类 , dealloc释放资源,私有方法 _isKVOA 是用来标示该类是一个 KVO 机制声称的类。在这里 swift 做了一些优化,它对所有被观察对象只生成一个派生类,该派生类实现所有被观察对象的 setter 方法,这样就减少了派生类的数量,提供了效率。所有 NSKVONotifying_TestKVOKVC.People 这个派生类重写了 setAge,setName方法(留意:没重写其他set方法)。
接着来看最后两行输出,地址 0x00000001083a3c60 是 TestKVOKVC.People 类中的实现,而地址是 0x00000001089807fa 是派生类 NSKVONotifying_TestKVOKVC.People 类中的实现。那后面那个地址到底是什么呢?可以通过 lldb 的 info 命令 image lookup --address 0x00000001089807fa 查看该地址的信息:
(lldb) image lookup --address 0x00000001089807fa
Address: Foundation[0x00000000000637fa] (Foundation.__TEXT.__text + 401690)
Summary: Foundation`_NSSetObjectValueAndNotify
看起来它是 Foundation 框架提供的私有函数:_NSSetObjectValueAndNotify。更进一步,我们来看看 Foundation 到底提供了哪些用于 KVO 的辅助函数。打开 terminal,使用 nm -a 命令查看 Foundation 中的信息:
nm -a /System/Library/Frameworks/Foundation.framework/Versions/C/Foundation
其中查找到我们关注的函数:
00000000000c76bf t __NSSetBoolValueAndNotify
0000000000152ab2 t __NSSetBoolValueForKeyInIvar
00000000000c74c8 t __NSSetBoolValueForKeyWithMethod
00000000000997db t __NSSetCharValueAndNotify
0000000000064759 t __NSSetCharValueForKeyInIvar
00000000000646f1 t __NSSetCharValueForKeyWithMethod
000000000001bd73 t __NSSetCheckSize
00000000003f7a50 b __NSSetClass
00000000001019af t __NSSetDirectory2
00000000000ab4d3 t __NSSetDoubleValueAndNotify
0000000000152b5d t __NSSetDoubleValueForKeyInIvar
00000000000abf65 t __NSSetDoubleValueForKeyWithMethod
0000000000139763 T __NSSetExceptionRaiser
0000000000180650 t __NSSetFileModificationUNIXTime
00000000000c9cca t __NSSetFloatValueAndNotify
0000000000152bb6 t __NSSetFloatValueForKeyInIvar
0000000000152712 t __NSSetFloatValueForKeyWithMethod
00000000000c9ba9 t __NSSetIntValueAndNotify
0000000000152c0f t __NSSetIntValueForKeyInIvar
00000000000abefb t __NSSetIntValueForKeyWithMethod
00000000001817a7 T __NSSetLogCStringFunction
00000000000a5389 t __NSSetLongLongValueAndNotify
0000000000152d5f t __NSSetLongLongValueForKeyInIvar
00000000000a55e1 t __NSSetLongLongValueForKeyWithMethod
000000000015abd7 t __NSSetLongValueAndNotify
0000000000152cb7 t __NSSetLongValueForKeyInIvar
0000000000152779 t __NSSetLongValueForKeyWithMethod
000000000011651f T __NSSetMainBundle
00000000001531ad t __NSSetNilValueForKey
00000000000647b0 t __NSSetObjectSetIvarValueForKeyInIvar
0000000000096e38 t __NSSetObjectValueAndNotify
0000000000028b2d t __NSSetObjectValueForKeyInIvar
00000000000c0657 t __NSSetPointValueAndNotify
0000000000152e5f t __NSSetPointValueForKeyInIvar
00000000000c05f1 t __NSSetPointValueForKeyWithMethod
000000000015b067 t __NSSetRangeValueAndNotify
0000000000152ebc t __NSSetRangeValueForKeyInIvar
00000000001528b2 t __NSSetRangeValueForKeyWithMethod
0000000000099610 t __NSSetRectValueAndNotify
0000000000152f15 t __NSSetRectValueForKeyInIvar
000000000015291d t __NSSetRectValueForKeyWithMethod
000000000015ae1f t __NSSetShortValueAndNotify
0000000000152db3 t __NSSetShortValueForKeyInIvar
0000000000152849 t __NSSetShortValueForKeyWithMethod
00000000000af692 t __NSSetSizeValueAndNotify
0000000000152f86 t __NSSetSizeValueForKeyInIvar
00000000000af62d t __NSSetSizeValueForKeyWithMethod
000000000015aab3 t __NSSetUnsignedCharValueAndNotify
0000000000152b09 t __NSSetUnsignedCharValueForKeyInIvar
00000000001526a9 t __NSSetUnsignedCharValueForKeyWithMethod
00000000000e59a2 t __NSSetUnsignedIntValueAndNotify
0000000000152c63 t __NSSetUnsignedIntValueForKeyInIvar
0000000000103f5e t __NSSetUnsignedIntValueForKeyWithMethod
0000000000096fa8 t __NSSetUnsignedLongLongValueAndNotify
00000000000f5f93 t __NSSetUnsignedLongLongValueForKeyInIvar
00000000000c745d t __NSSetUnsignedLongLongValueForKeyWithMethod
000000000015acfa t __NSSetUnsignedLongValueAndNotify
0000000000152d0b t __NSSetUnsignedLongValueForKeyInIvar
00000000001527e1 t __NSSetUnsignedLongValueForKeyWithMethod
000000000015af43 t __NSSetUnsignedShortValueAndNotify
0000000000152e09 t __NSSetUnsignedShortValueForKeyInIvar
0000000000103ef5 t __NSSetUnsignedShortValueForKeyWithMethod
Foundation 提供了大部分基础数据类型的辅助函数,此外还包括一些常见的 Cocoa 结构体如 Point, Range, Rect, Size,这表明这些结构体也可以用于自动键值观察,但要注意除此之外的结构体就不能用于自动键值观察了。
待续...........
demo:https://github.com/easyui/TestKVO
参考:
http://blog.csdn.net/wzzvictory/article/details/9674431?utm_source=tuicool
http://blog.csdn.net/kesalin/article/details/8194240
http://www.cnblogs.com/dark-angel/archive/2011/05/05/2037734.html
http://objccn.io/issue-7-3/#key-value-validation
http://www.cnblogs.com/496668219long/p/4470923.html
http://www.jianshu.com/p/e036e53d240e
http://blog.sina.com.cn/s/blog_8a38e5240100u2yw.html
http://www.androiddev.net/kvo/
http://nshipster.com/key-value-observing/
http://tech.glowing.com/cn/implement-kvo/
https://developer.apple.com/library/mac/documentation/Cocoa/Conceptual/KeyValueObserving/Articles/KVOImplementation.html
相关推荐
Swift5.2 Key-Value Observing (KVO) 是一种Objective-C中的机制,但在Swift中也可以使用,主要是为了实现对象属性变化的监听。KVO允许一个对象观察另一个对象的某个属性,当该属性的值发生变化时,观察者会收到通知...
在Swift编程中,KVO(Key-ValueObserving,键值观察)是一种强大的机制,用于监听对象属性的变化。本文将详细讲解如何在Swift中利用`willMoveToSuperview`方法无痛地实现KVO,以及其在Swift开发中的应用。 ### 1. ...
**iOS KVO(Key-Value Observing)实现原理** Key-Value Observing,简称KVO,是Objective-C中的一种机制,允许对象监听其他对象的属性变化。KVO是Apple为Foundation框架提供的一种强大的数据绑定技术,它使得一个...
Swift中的Key-Value Coding(KVC)和Key-Value Observing(KVO)是两种强大的数据绑定技术,它们允许程序员间接访问对象的属性并监听这些属性的变化。在本文中,我们将深入探讨这两种技术的使用方法、工作原理以及...
在Swift编程中,Key-Value Observing(KVO)是一种观察者模式的实现,用于监听对象属性的变化。KVO在Objective-C中广泛使用,并且在Swift中也可以通过桥接头文件来实现。标题提到的“swift-一句话使用KVO使用完无需...
**KVO(Key-ValueObserving)实现原理详解** KVO,全称为Key-ValueObserving,是Objective-C中一种强大的数据绑定机制,允许对象监听并自动响应其他对象属性值的变化。KVO是Apple在Foundation框架中提供的一个API,...
在iOS开发中,Key-Value Observing(KVO)是一种强大的机制,允许对象监听其他对象属性的变化。在标题“iOS KVO 监听frame”中,我们关注的是如何使用KVO来跟踪和响应UI控件(如UIView)的frame属性变化。这在布局...
### KVC、kVO原理详解 #### 一、KVC(Key-Value Coding)与 KVO(Key-Value Observing)简介 KVC 和 KVO 是 Objective-C 中两个非常重要的特性,它们为开发者提供了强大的数据访问和观察机制。这两种技术在 Cocoa ...
Swift-Ausbin是一个基于数据驱动的Swift框架,其核心特性是采用了Key-Value Observing (KVO)机制。KVO是Objective-C中的一个强大的特性,用于观察并响应对象属性值的变化。在Swift中,虽然没有原生支持KVO,但通过...
详细的介绍了KVO与KVC的原理机制,可以有效运用KVO-KVC编程
**KVO机制工作原理** Key-Value Observing(KVO)是Objective-C中的一种机制,它允许对象监听并响应其他对象属性的变化。这种机制在iOS和macOS开发中广泛用于数据绑定和实时更新视图。KVO是Apple的Cocoa框架的一...
Swift-KVO,即Key-Value Observing,是iOS开发中的一种数据绑定技术,它允许对象监听其他对象的属性变化。在Objective-C中,KVO是一个强大的特性,而在Swift中,虽然原生不支持,但可以通过一些技巧实现。这篇文章将...
首先,我们需要理解KVO的基本原理。KVO是Objective-C的一部分,但在Swift中也可以通过桥接头文件使用。当一个对象注册为另一个对象的观察者时,它会接收到被观察对象属性变化的通知。通常,我们使用`addObserver...
Swift5.2中的Key-Value Observing(KVO)是一种观察者模式的实现,它允许一个对象监听并响应另一个对象的属性变化。在iOS、macOS等Apple平台的开发中,KVO是一种强大的机制,可以帮助我们实现数据绑定和动态更新视图...
在Swift编程语言中,Key-ValueObserving (KVO)是一种强大的机制,允许我们监听某个对象的属性变化。虽然Objective-C原生支持KVO,但Swift并没有直接提供类似的功能。然而,开发者可以通过自定义实现来达到类似的效果...
在iOS开发中,Swift虽然原生并不支持响应式编程,但我们可以通过KVO(Key-Value Observing)或者第三方库如RxSwift、ReactiveCocoa来实现类似的功能。 KVO,即Key-Value Observing,是Objective-C中的一个特性,但...
这个主题“Swift-KVO+Runtime转换模型”涉及到Swift中的Key-Value Observing (KVO)和Runtime机制,它们是Objective-C的特性,但在Swift中也可以通过桥接头文件使用。 **Key-Value Observing (KVO)** 是Objective-C...
2. **KVO基本原理** KVO的实现依赖于Objective-C的isa指针和消息发送机制。当一个对象被添加到观察者列表后,系统会动态地生成一个新的类,这个类继承自原始类,并重写了被观察属性的setter方法,以实现通知观察者...
iOS开发·KVO用法,原理与底层实现: runtime模拟实现KVO监听机制(Blcok及Delgate方式).zip,iOS开发·KVO用法,原理与底层实现: runtime模拟实现KVO监听机制(Blcok及Delgate方式)
正在开发的App为金融类型,后台频繁快速推送数据,UI界面需要及时响应。为了解决频繁reload UITableView等控件,解决办法是某个数据发生变化,仅刷新相应的独立UI元素(如,UILable,UIButton)。