【iOS】KVO&KVC原理

1 KVO 键值监听

1.1 KVO简介

KVO的全称是Key-Value Observing,俗称"键值监听",可以用于监听摸个对象属性值得改变。

KVO一般通过以下三个步骤使用:

objectivec 复制代码
// 1. 添加监听
[self.student1 addObserver:self forKeyPath:@"age" options:options context:nil];


// 2. 重写- (void)observeValueForKeyPath:(NSString *)keyPath ofObject:(id)object change:(NSDictionary<NSKeyValueChangeKey,id> *)change context:(void *)context方法

- (void)observeValueForKeyPath:(NSString *)keyPath ofObject:(id)object change:(NSDictionary<NSKeyValueChangeKey,id> *)change context:(void *)context {
    NSLog(@"%@的%@被改变:%@", object, keyPath, change);
}

// 3. 适当时机移除监听
[self.student1 removeObserver:self forKeyPath:@"age"];

1.2 KVO简单使用

  1. 建立SXStudent类和SXTeacher
objectivec 复制代码
//SXStudent.h 

#import <Foundation/Foundation.h>

NS_ASSUME_NONNULL_BEGIN

@interface SXStudent : NSObject
@property (nonatomic, assign) NSInteger age;
@end

NS_ASSUME_NONNULL_END


// SXTeacher.h
#import <Foundation/Foundation.h>
#import "SXStudent.h"

NS_ASSUME_NONNULL_BEGIN

@interface SXTeacher : NSObject
@property (nonatomic, strong) SXStudent *student1;
@property (nonatomic, strong) SXStudent *student2;
- (void)demo;
@end

NS_ASSUME_NONNULL_END
  1. 实现SXStudent类。
objectivec 复制代码
// SXStudent.m

#import "SXStudent.h"
@implementation SXStudent
@end
  1. 实现SXTeacher类,重写init方法,为SXTeacherstudent1属性添加监听。实现demo方法,分别更改student1student2age值。
objectivec 复制代码
// SXTeacher.m

#import "SXTeacher.h"
#import <objc/runtime.h>

@implementation SXTeacher

- (id)init {
    if (self = [super init]) {
        self.student1 = [[SXStudent alloc] init];
        self.student2 = [[SXStudent alloc] init];
        
        self.student1.age = 1;
        self.student2.age = 2;
        
        // 添加监听
        NSKeyValueObservingOptions options = NSKeyValueObservingOptionNew | NSKeyValueObservingOptionOld;
        [self.student1 addObserver:self forKeyPath:@"age" options:options context:nil];
    }
    return self;
}

- (void)demo {
    self.student1.age = 20;
    self.student2.age = 30;
}

- (void)observeValueForKeyPath:(NSString *)keyPath ofObject:(id)object change:(NSDictionary<NSKeyValueChangeKey,id> *)change context:(void *)context {
    NSLog(@"%@的%@被改变:%@", object, keyPath, change);
}

- (void)dealloc {
    // 移除监听
    [self.student1 removeObserver:self forKeyPath:@"age"];
}

@end
  1. mian函数内创建SXTeacher的实例对象并调用demo方法测试。
objectivec 复制代码
#import <Foundation/Foundation.h>
#import "SXTeacher.h"

int main(int argc, const char * argv[]) {
    @autoreleasepool {
        SXTeacher *teacher = [[SXTeacher alloc] init];
        [teacher demo];
    }
    return 0;
}
  1. 查看运行结果。

1.3 实现原理探究

1.3.1 student1发生的变化

为什么student1setter方法可以触发监听,添加监听的方法到底对student1做了什么?

  1. 我们在添加监听后打一个断点。
  2. 试着利用lldb调试查看student1student2isa指针。

我们发现student1isa指针的指向被更改成了NSKVONotifying_SXStudent(NSKVONotifying_为前缀,原类名为后缀)类。

1.3.2 NSKVONotifying_XXX类

  1. 关于NSKVONotifying_XXX
  • NSKVONotifying_XXX类是 Runtime动态创建的一个类,在程序运行的过程中产生的一个新的类。
  • NSKVONotifying_XXX类是原类的一个子类。
  • NSKVONotifying_XXX类存在自己的 setAge:classdeallocisKVOA...方法。

试着验证NSKVONotifying_XXX类的方法和父类,我们可以使用如下代码打印NSKVONotifying_SXStudent类和SXStudent类的方法列表和父类类型。

objectivec 复制代码
- (void)demo2 {
    [self printMethods:object_getClass(self.student1)];
    [self printMethods:object_getClass(self.student2)];
}

- (void) printMethods:(Class)cls {
    unsigned int count;
    Method *methods = class_copyMethodList(cls, &count);
    NSMutableString *methodNames = [NSMutableString string];
    [methodNames appendFormat:@"%@ - ", cls];
    NSLog(@"%@ superClass ----> %@", NSStringFromClass(cls), NSStringFromClass(class_getSuperclass(cls)));
    
    for (int i = 0; i < count; i++) {
        Method method = methods[i];
        NSString *methodName = NSStringFromSelector(method_getName(method));
        
        [methodNames appendFormat:@"%@ ", methodName];
    }
    
    NSLog(@"%@", methodNames);
    free(methods);
}

打印结果:

可以看到NSKVONotifying_SXStudent类有自己的setAge:classdealloc_isKVOA方法。

  • 重写class方法是为了隐藏NSKVONotifying_XXX类的存在。重写后的class方法返回其父类(原来的类)类型,使用户以为类没有变化。
  • _isKVOA则是用来标识当前类是否是通过runtime动态生成的类对象,如果是,就返回YES,不是,则返回NO。
  • 当对像被销毁后,dealloc做一些收尾工作。

1.3.3 方法调用探究

由上面可分析出我们的student1isa指针指向的类对象是NSKVONotifying_SXStudent,并且NSKVONotifying_SXStudent中还带有setAge: 方法,所以student1setAge:方法走的应该是NSKVONotifying_SXStudent类中的setAge:方法。

我们试着使用下面的代码打印student1被监听前后的setAge:方法的地址,并使用lldb调试一探究竟。

objectivec 复制代码
- (id)init {
    if (self = [super init]) {
        self.student1 = [[SXStudent alloc] init];
        self.student2 = [[SXStudent alloc] init];
        
        self.student1.age = 1;
        self.student2.age = 2;
        
        NSLog(@"添加监听之前 - p1 = %p, p2 = %p", [self.student1 methodForSelector:@selector(setAge:)], [self.student2 methodForSelector:@selector(setAge:)]);
        
        // 添加监听
        NSKeyValueObservingOptions options = NSKeyValueObservingOptionNew | NSKeyValueObservingOptionOld;
        [self.student1 addObserver:self forKeyPath:@"age" options:options context:nil];
        
        NSLog(@"添加监听之后 - p1 = %p, p2 = %p", [self.student1 methodForSelector:@selector(setAge:)], [self.student2 methodForSelector:@selector(setAge:)]);
    }
    return self;
}

打断点使用lldb打印方法地址对应的方法名:

我们发现student1setAge:方法实际上是调用了Foundation框架的_NSSetLongLongValueAndNotify函数。这又是怎么回事,我们先来了解一下这个函数。

1.3.3 _NSSetXXXValueAndNotify

经过查阅资料我们可以了解到。
NSKVONotifyin_XXX中的setage:方法中其实调用了 Fundation框架中C语言函数 _NSsetXXXValueAndNotify_NSsetXXXValueAndNotify内部做的操作相当于,首先调用willChangeValueForKey 将要改变方法,之后调用原来的setage方法对成员变量赋值,最后调用didChangeValueForKey已经改变方法。didChangeValueForKey中会调用监听器的监听方法,最终来到监听者的observeValueForKeyPath方法中。

Foundation框架中会根据属性的类型,调用不同的方法。例如我们之前定义的NSInteger类型的age属性,那么我们看到Foundation框架中调用的_NSsetLongLongValueAndNotify函数。那么我们把age的属性类型变为double重新打印一遍。

我们发现调用的函数变为了_NSSetDoubleValueAndNotify,那么这说明Foundation框架中有许多此类型的函数,通过属性的不同类型调用不同的函数。

我们可以重写 SXStudent类的willChangeValueForKey方法和didChangeValueForKey方法来验证上述说法。

objectivec 复制代码
#import "SXStudent.h"

@implementation SXStudent
- (void)setAge:(NSInteger)age {
    NSLog(@"setAge");
    _age = age;
}
- (void)willChangeValueForKey:(NSString *)key {
    NSLog(@"willChangeValueForKey begin");
    [super willChangeValueForKey:key];
    NSLog(@"willChangeValueForKey end");
}

- (void)didChangeValueForKey:(NSString *)key {
    NSLog(@"didChangeValueForKey begin");
    [super didChangeValueForKey:key];
    NSLog(@"didChangeValueForKey end");
}
@end

打印结果:

可知:

  • _NSSetXXXValueAndNotify调用willChangeValueForKey:
  • _NSSetXXXValueAndNotify调用setter实现;
  • _NSSetXXXValueAndNotify调用didChangeValueForKey:
  • didChangeValueForKey内部会调用observer的observeValueForKeyPath:ofObject:change:context:方法。

1.3.4 伪代码

据上所述,可以写出NSKVONotifying_SXStudent类的伪代码:

objectivec 复制代码
///> NSKVONotifying_SXStudent.m 文件

#import "NSKVONotifying_SXStudent.h"

@implementation NSKVONotifying_SXStudent

- (void)setAge:(int)age{
  _NSSetLongLongValueAndNotify();  ///> 文章末尾 知识点补充小结有此方法来源
}

void _NSSetLongLongValueAndNotify(){
  [self willChangeValueForKey:@"age"];
  [super setAge:age];
  [self didChangeValueForKey:@"age"];
}

- (void)didChangeValueForKey:(NSString *)key{
  ///> 通知监听器 key发生了改变
  [observe observeValueForKeyPath:key ofObject:self change:nil context:nil];
}

@end

2 KVC 键值编码

2.1 KVC简介

KVC的全称key - value - coding,俗称"键值编码",可以通过key来访问某个属性。

常见的API有:

objectivec 复制代码
- (void)setValue:(id)value forKey:(NSString *)key;
- (void)setValue:(id)value forKeyPath:(NSString *)keyPath;

- (id)valueForKey:(NSString *)key 
- (id)valueForKeyPath:(NSString *)keyPath;

2.2 KVC简单使用

2.2.1 自定义SXDog类、SXStudent类和SXTeacher类。

objectivec 复制代码
// SXDog.h
#import <Foundation/Foundation.h>

NS_ASSUME_NONNULL_BEGIN

@interface SXDog : NSObject
@property (nonatomic, assign) CGFloat weight;
@end

NS_ASSUME_NONNULL_END


// SXStudent.h
#import <Foundation/Foundation.h>
#import "SXDog.h"

NS_ASSUME_NONNULL_BEGIN

@interface SXStudent : NSObject
@property (nonatomic, assign) NSInteger age;
@property (nonatomic, strong) SXDog *dog;
@end

NS_ASSUME_NONNULL_END


// SXTeacher.h
#import <Foundation/Foundation.h>
#import "SXStudent.h"

NS_ASSUME_NONNULL_BEGIN

@interface SXTeacher : NSObject
@property (nonatomic, strong) SXStudent *student1;
@end

NS_ASSUME_NONNULL_END

2.2.2 实现这三个类,为了方便使用,重写SXStudent和SXTeacher的初始化方法,在初始化方法里对属性进行初始化。

SXDog.m

objectivec 复制代码
#import "SXDog.h"

@implementation SXDog
@end

SXStudent.m

objectivec 复制代码
#import "SXStudent.h"
#import <objc/runtime.h>

@implementation SXStudent
- (id)init {
    if (self = [super init]) {
        self.dog = [[SXDog alloc] init];
    }
    return self;
}
@end

SXTeacher.m

objectivec 复制代码
#import "SXTeacher.h"
#import <objc/runtime.h>

@implementation SXTeacher
- (id)init {
    if (self = [super init]) {
        self.student1 = [[SXStudent alloc] init];
    }
    return self;
}
@end

2.2.3 为SXTeacher添加两个方法,在这两个方法里试调用KVC。

  1. SetValue:ForKey:ValueForKey:方法
objectivec 复制代码
- (void)demoSetValueForKeyAndValueForKey {
    [self.student1 setValue:@20 forKey:@"age"];
    NSLog(@"点语法:%ld", self.student1.age);
    NSNumber *value = [self.student1 valueForKey:@"age"];
    NSLog(@"KVC:%@", value);
}
  1. SetValue:ForKeyPath:ValueForKeyPath:
objectivec 复制代码
- (void)demoSetValueForKeyPathAndValueForKeyPath {
    [self.student1 setValue:@16 forKeyPath:@"dog.weight"];
    NSLog(@"点语法:%lf", self.student1.dog.weight);
    NSNumber *value = [self.student1 valueForKeyPath:@"dog.weight"];
    NSLog(@"KVC:%@", value);
}
  1. 调用上面两个函数,运行。

2.2.4 KeyPath 和 Key 的区别:

  • keyPath 相当于根据路径去寻找属性,一层一层往下找。
  • key 是直接访问属性的名字,如果按路径找会报错。

2.3 KVC流程

2.3.1 setValue:forkey:赋值流程

  • 首先会按照setKey:、_setKey:的顺序到对象的方法列表中寻找这两个方法,如果找到了方法,则传参并且调用方法。
  • 如果没有找到方法,则通过accessInstanceVariablesDirectly方法的返回值来决定是否能够查找成员变量。如果accessInstanceVariablesDirectly返回YES,则会按照以下顺序到成员变量列表中查找对应的成员变量:
    • _key
    • _isKey
    • key
    • isKey
  • 如果accessInstanceVariablesDirectly返回NO,则直接抛出NSUnknownKeyException异常。
    如果在成员变量列表中找到对应的属性值,则直接进行赋值,如果找不到,则会抛出NSUnknownKeyException异常。

accessInstanceVariablesDirectly函数

objectivec 复制代码
+ (BOOL)accessInstanceVariablesDirectly{
      return YES;   ///> 可以直接访问成员变量
  //    return NO;  ///>  不可以直接访问成员变量,  
  ///> 直接访问会报NSUnkonwKeyException错误  
}

2.3.2 valueForKey:取值流程

  • 首先会按照以下顺序查找方法列表:
    • getKey
    • key
    • isKey
    • _key
  • 如果找到就直接传递参数,调用方法,如果未找到则查看accessInstanceVariablesDirectly方法的返回值,如果返回NO,则直接抛出NSUnknownKeyException异常。
  • 如果accessInstanceVariablesDirectly方法返回YES,则按如下顺序查找成员变量列表:
    • _key
    • _isKey
    • key
    • isKey
  • 如果能找到对应的成员变量,则直接获取成员变量的值,如果未找到,则抛出NSUnknownKeyException异常。

2.3.3 试验证setValue:forkey:赋值流程

对上述例子进行小修改:

SXStudent.h

objectivec 复制代码
@interface SXStudent : NSObject {
    @public
    int _age;
    int _isAge;
    int age;
    int isAge;
}
@end

SXTeacher.m

objectivec 复制代码
- (id)init {
    if (self = [super init]) {
        self.student1 = [[SXStudent alloc] init];
        
        NSKeyValueObservingOptions option = NSKeyValueObservingOptionNew | NSKeyValueObservingOptionOld;
        [self.student1 addObserver:self forKeyPath:@"age" options:option context:nil];
        
        [self.student1 setValue:@20 forKey:@"age"];
        
        NSLog(@"-----");
    }
    return self;
}
- (void)observeValueForKeyPath:(NSString *)keyPath ofObject:(id)object change:(NSDictionary<NSKeyValueChangeKey,id> *)change context:(void *)context {
    NSLog(@"%@的%@被改变:%@", object, keyPath, change);
}

NSLog(@"-----");处打下断点,运行,查看student1中的成员变量。看看谁被赋值了。

可以看到_age首先被赋值,我们注释掉SXStudent中的_age成员变量,看看下一个是谁被赋值。如此反复,就可以得到setValue:forkey:赋值流程。结果与上述无误,我就不继续了。

通过本例,我们还可以知道KVC也可以触发KVO监听。

3 一些问题

3.1 iOS用什么方式实现对一个对象的KVO?(KVO的本质是什么?)

给一个实例对象添加KVO,系统内部是利用Runtime动态的生成一个此实例对象的类对象的子类,具体的格式为_NSKVONotifying_XXX,并且让实例对象的isa指针指向这个新生成的类。

重写属性的set方法,当调用set方法时,会调用Foundation框架的NSSetXXXValueAndNotify函数

_NSSetXXXValueAndNotify中会执行以下步骤:

  • 调用willChangeValueForKey:方法;
  • 调用父类的set方法,重新赋值;
  • 调用didChangeValueForKey:方法;
  • didChangeValueForKey:内部会触发监听器的observeValueForKeyPath:ofObject:change:context:方法。

3.2 如何手动触发KVO?

手动调用willChangeValueForKey:didChangeValueForKey:

例:

objectivec 复制代码
- (id)init {
    if (self = [super init]) {
        self.student1 = [[SXStudent alloc] init];
        
        NSKeyValueObservingOptions option = NSKeyValueObservingOptionNew | NSKeyValueObservingOptionOld;
        [self.student1 addObserver:self forKeyPath:@"age" options:option context:nil];
        [self.student1 willChangeValueForKey:@"age"];
        [self.student1 didChangeValueForKey:@"age"];
        
    }
    return self;
}

运行结果:

虽然是在didChangeValueForKey:内部会触发监听器的observeValueForKeyPath:ofObject:change:context:方法,但是如果不调用willChangeValueForKey:无法就无法触发监听器,这两个必须一起使用。

3.3 直接修改成员变量的值是否会触发KVO?

直接修改成员变量的值不会触发KVO,因为没有触发setter方法。

相关推荐
用户0916 小时前
SwiftUI Charts 函数绘图完全指南
ios·swiftui·swift
YungFan16 小时前
iOS26适配指南之UIColor
ios·swift
权咚1 天前
阿权的开发经验小集
git·ios·xcode
用户091 天前
TipKit与CloudKit同步完全指南
ios·swift
小溪彼岸1 天前
macOS自带截图命令ScreenCapture
macos
法的空间2 天前
Flutter JsonToDart 支持 JsonSchema
android·flutter·ios
2501_915918412 天前
iOS 上架全流程指南 iOS 应用发布步骤、App Store 上架流程、uni-app 打包上传 ipa 与审核实战经验分享
android·ios·小程序·uni-app·cocoa·iphone·webview
TESmart碲视2 天前
Mac 真正多显示器支持:TESmart USB-C KVM(搭载 DisplayLink 技术)如何实现
macos·计算机外设·电脑
00后程序员张2 天前
iOS App 混淆与加固对比 源码混淆与ipa文件混淆的区别、iOS代码保护与应用安全场景最佳实践
android·安全·ios·小程序·uni-app·iphone·webview
Magnetic_h2 天前
【iOS】设计模式复习
笔记·学习·ios·设计模式·objective-c·cocoa