iOS底层探究iOS开发

iOS中KVO的底层实现原理

2020-01-05  本文已影响0人  一叶知秋0830

1. KVO的使用

KVO(Key-Value Observing),也就是我们常说的键值监听,可以用于监听某个对象属性值的改变。KVO使用比较简单,如下所示定义了一个含有2个属性的Student类,然后声明一个实例对象,并添加一个观察者监听某个属性,当被监听的属性发生变化时就会调用观察者的observeValueForKeyPath: ofObject: change: context:方法。当不需要监听的时候需要移除观察者。

// Student.h文件
@interface Student : NSObject
@property (nonatomic , strong) NSString *name;
@property (nonatomic , strong) NSMutableArray booksArr;
@end
// 使用Student类的文件
- (void)test{
    self.stu1 = [[Student alloc] init];
    // 添加观察者监听name的变化
    [self.stu1 addObserver:self
                forKeyPath:@"name"
                   options:NSKeyValueObservingOptionNew | NSKeyValueObservingOptionOld
                   context:NULL];

    NSLog(@"name改变前");
    self.stu1.name = @"Jack";
    NSLog(@"name改变后");
}

// 当监听属性发生变化时的回调
- (void)observeValueForKeyPath:(NSString *)keyPath ofObject:(id)object change:(NSDictionary<NSKeyValueChangeKey,id> *)change context:(void *)context{
    NSLog(@"keyPath:%@,change-->%@",keyPath,change);
}

- (void)dealloc{
    // 移除观察者
    [self.stu1 removeObserver:self forKeyPath:@"name"];
}

// ********************打印结果********************
2020-01-05 09:42:32.371008+0800 GCDDemo[13375:567451] name改变前
2020-01-05 09:42:32.371618+0800 GCDDemo[13375:567451] keyPath:name,change-->{
    kind = 1;
    new = Jack;
    old = "<null>";
}
2020-01-05 09:42:32.371895+0800 GCDDemo[13375:567451] name改变后

2. KVO底层实现原理

KVO的实现过程实际上是利用了OC的runtime机制,当一个实例对象(比如上面的self.stu1)添加观察者时,底层根据该实例对象所属的类动态添加了一个类(动态添加的类名就是在原来类的类名前加上NSKVONotifying_前缀),这个类是继承自原来的类的。上面实例的底层实现过程如下:

Foundation框架下的_NSSet***ValueAndNotify系列方法列表如下:

 _NSSetBoolValueAndNotify 
 _NSSetCharValueAndNotify 
 _NSSetDoubleValueAndNotify 
 _NSSetFloatValueAndNotify 
 _NSSetIntValueAndNotify 
 _NSSetLongLongValueAndNotify 
 _NSSetLongValueAndNotify 
 _NSSetObjectValueAndNotify 
 _NSSetPointValueAndNotify 
 _NSSetRangeValueAndNotify 
 _NSSetRectValueAndNotify 
 _NSSetShortValueAndNotify 
 _NSSetSizeValueAndNotify 
 _NSSetUnsignedCharValueAndNotify 
 _NSSetUnsignedIntValueAndNotify 
 _NSSetUnsignedLongLongValueAndNotify 
 _NSSetUnsignedLongValueAndNotify 
 _NSSetUnsignedShortValueAndNotify 

3. KVO底层实现的验证

3.1 我们怎么知道添加观察者时动态添加了一个类?

这个其实我们只需要打印一下再添加观察者之前和之后实例对象所属的类就知道了。不过前面已经说过了,动态添加的类重写了class方法,所以我们不能通过这个方法来获取一个实例对象的类,而要通过runtimeobject_getClass()这个API来获取:

- (void)test1{
    self.stu1 = [[Student alloc] init];
    
    NSLog(@"观察前- [self.stu1 class] -->%@",[self.stu1 class]);
    NSLog(@"观察前- object_getClass(self.stu1) -->%@",object_getClass(self.stu1));

    [self.stu1 addObserver:self forKeyPath:@"name" options:NSKeyValueObservingOptionNew | NSKeyValueObservingOptionOld context:NULL];

    NSLog(@"观察后- [self.stu1 class] -->%@",[self.stu1 class]);
    NSLog(@"观察后- object_getClass(self.stu1) -->%@",object_getClass(self.stu1));
}

// ********************打印结果********************
2020-01-05 10:51:00.584299+0800 GCDDemo[14497:600230] 观察前- [self.stu1 class] -->Student
2020-01-05 10:51:00.584690+0800 GCDDemo[14497:600230] 观察前- object_getClass(self.stu1) -->Student
2020-01-05 10:51:00.592797+0800 GCDDemo[14497:600230] 观察后- [self.stu1 class] -->Student
2020-01-05 10:51:00.593064+0800 GCDDemo[14497:600230] 观察后- object_getClass(self.stu1) -->NSKVONotifying_Student

3.2 如何知道重写了哪些方法?

这里我们需要用到runtime的一些API来获取一个类对象里面存储的方法列表信息,下面我们先封装一个方法来获取这些信息,然后把监听前和监听后的方法列表打印出来。

- (void)test2{
    self.stu1 = [[Student alloc] init];

    NSLog(@"观察前方法列表-->%@",[self methodNamesOfClass:object_getClass(self.stu1)]);

    [self.stu1 addObserver:self forKeyPath:@"name" options:NSKeyValueObservingOptionNew | NSKeyValueObservingOptionOld context:NULL];

    NSLog(@"观察后方法列表-->%@",[self methodNamesOfClass:object_getClass(self.stu1)]);
  
}

// 传入一个类,将类中方法列表的方法名拼接换成字符串返回
- (NSString *)methodNamesOfClass:(Class)cls{
    unsigned int count;
    // 获取方法列表
    Method *methodList = class_copyMethodList(cls, &count);
    NSString *methodNamesStr = @"";

    // 遍历方法列表将方法名拼接成字符串
    for (int i = 0; i < count; i++) {
        Method method = methodList[i];
        NSString *methodName = NSStringFromSelector(method_getName(method));
        methodNamesStr = [methodNamesStr stringByAppendingFormat:@"%@ ,",methodName];
    }
    
    // 释放
    free(methodList);
    
    return methodNamesStr;
}

// ********************打印结果********************
2020-01-05 10:56:43.077817+0800 GCDDemo[14606:603376] 观察前方法列表-->.cxx_destruct ,name ,setName: ,age ,setAge: ,
2020-01-05 10:56:43.078483+0800 GCDDemo[14606:603376] 观察后方法列表-->setName: ,class ,dealloc ,_isKVOA ,

3.3 怎么知道重写setter方法是调用的哪个方法?

这里我们同样需要用到runtime的API,首先通过class_getInstanceMethod()函数来获取setter方法的Method,然后再调用method_getImplementation()来得到setter方法的IMP

不过我们首先打印的是IMP的地址,想要看IMP的具体信息我们需要打一个断点调出LLDB,然后借助LLDB来打印具体信息。比如在监听前的IMP地址是0x10967d4c0,就可以在LLDB中输入p (IMP)0x10967d4c0来打印具体信息。从下面可以看出监听前setter方法就是正常的,监听后就变成了_NSSetObjectValueAndNotify

- (void)test1{
    self.stu1 = [[Student alloc] init];

    NSLog(@"监听前的setter方法IMP-->%p",[self IMPWithSelector:@selector(setName:)]);

    [self.stu1 addObserver:self forKeyPath:@"name" options:NSKeyValueObservingOptionNew | NSKeyValueObservingOptionOld context:NULL];

    NSLog(@"监听后的setter方法IMP-->%p",[self IMPWithSelector:@selector(setName:)]);
}

// 获取一个方法的IMP
- (IMP)IMPWithSelector:(SEL)selector{
    Class cls = object_getClass(self.stu1);
    Method methon = class_getInstanceMethod(cls, selector);
    IMP imp = method_getImplementation(methon);
    return imp;
}

// ********************打印结果********************
2020-01-05 11:25:40.485792+0800 GCDDemo[15032:617260] 监听前的setter方法IMP-->0x10967d4c0
2020-01-05 11:25:40.489656+0800 GCDDemo[15032:617260] 监听后的setter方法IMP-->0x7fff25701c8a
(lldb) p (IMP)0x10967d4c0
(IMP) $0 = 0x000000010967d4c0 (GCDDemo`-[Student setName:] at Student.h:15)
(lldb) p (IMP)0x7fff25701c8a
(IMP) $1 = 0x00007fff25701c8a (Foundation`_NSSetObjectValueAndNotify)

4. KVO小结

KVO的核心是动态生成一个继承自原类的类,然后将实例对象的isa指向这个类。然后重写了监听属性的setter方法,在原有setter方法的前面调用willChangeValueForKey方法,在原有setter方法的后面调用didChangeValueForKey

所以我们要判断某个操作是否会触发KVO关键在于它是否调用了监听属性的setter方法。比如上面的例子,self.stu1.name = @"Jack";这种方式就是调用setter方法,所以它会触发KVO。但是下面这几种方式是不会触发KVO的:

上面这几种情况,如果我们也想触发KVO的话,我们可以手动触发,也就是在原有方法的前面和后面分别加上willChangeValueForKeydidChangeValueForKey这两个方法。就比如最后这个例子,我们可以这样写:

[self.stu1 willChangeValueForKey:@"dog"];
self.stu1.dog.age = 3;
[self.stu1 didChangeValueForKey:@"dog"];

最后还有一点要说明,通过KVC方式设置属性值也是会触发KVO的。比如[self.stu1 setValue:@"Jack" forKey:@"name"];这样写是可以触发KVO的,这应该是苹果在KVC实现中调用了willChangeValueForKeydidChangeValueForKey这两个方法。

上一篇 下一篇

猜你喜欢

热点阅读