runtime 实现 kvo

KVO的核心就是动态的修改了原来类的set方法

先来回顾一下KVO的具体用法:
首先创建一个person类,给一个name属性。

@interface Person : NSObject
@property (nonatomic,copy) NSString *name;
@end

然后添加控制器为监察者

    Person *p = [[Person alloc] init];
    //content 用来传入一个数值区分是否是同一个监察者。
    /* 当你重写这个方法的时候,不能确定父类或者子类是否也要用这个方法,此时可以使用不同content去区分。 */
    [p addObserver:self forKeyPath:@"name" options:NSKeyValueObservingOptionNew context:nil];
    _p = p;

实现监察代理方法

-(void)observeValueForKeyPath:(NSString *)keyPath ofObject:(id)object change:(NSDictionary<NSKeyValueChangeKey,id> *)change context:(void *)context {
    NSLog(@"名字是%@", _p.name);
}

最后记得removeObserver就OK了

开始自定义KVO

首先创建一个object的分类。添加一个addObserver的方法。

-(void)GYZ_addObserver:(NSObject *)observer forKeyPath:(NSString *)keyPath options:(NSKeyValueObservingOptions)options context:(void *)context {
    //第一步
    SEL setterSelector = NSSelectorFromString(setterForGetter(keyPath));
    Method setterMethod = class_getInstanceMethod([self class], setterSelector);
    if (!setterMethod) {
        // throw invalid argument exception
    }
   
    //第二步
    //1.1拼接类名
    NSString *oldClassName = NSStringFromClass([self class]);
    NSString *newClassName = [@"GYZ" stringByAppendingString:oldClassName];
    //1.2创建类
    Class MyClass = objc_allocateClassPair([self class], newClassName.UTF8String, 0);
    //2.添加set方法--重写!!
    /* 为什么要重写set方法, 因为虽然myClass继承自self,可以遵循继承体系去调用
        self的setname方法,
        但是集成体系把具体的调用给隐藏了,
        实际上是myclass的父类self本身去调用的自己的setname方法,myclass本身不能去调用。 */
    class_addMethod(MyClass, @selector(setName:), (IMP)setName, "v@:@");
    //注册类
    objc_registerClassPair(MyClass);
    //3.修改观察者的isa指针(将原有类的set方法调用改成了当前子类调用自己的set方法)
    object_setClass(self, MyClass);
}
//将key 转化为set方法的编号 setKey:  
NSString * setterForGetter(NSString *key) {
    NSString *newKey = [[[[key substringToIndex:1] uppercaseString] stringByAppendingString:[key substringFromIndex:1]] stringByAppendingString:@":"];
    return newKey;
}

第一步,对传进来的方法名字 keyPath 处理,加上set和:,然后根据这个编号去查找被观察者是否实现了这个set方法,如果没有实现,就不能使用kvo去监察。直接抛出异常。因为KVO监察的对象这能是可以被点语法调用的属性,必须走set方法。

第二步:1.动态生成一个类(为什么要生成一个类:这个类继承自被观察对象的类,只有生成这个子类,然后重写了原来类的set方法,才能够捕获到原有类的set方法调用前后,只有这样才能动态的通知所有观察这个对象的类。 最后会把原来类的isa指针指向这个新创建的子类,然后原来被观察的对象就变成了这个新的子类的实例对象。)

第三步,重写 setter 方法。新的 setter 在调用原 setter 方法后,通知每个观察者。 这里set方法的重写因人而异,可以根据自己的需要传出不同的数值,可以使用block、通知或者代理都行。

void setName(id self, SEL _cmd, NSString *str) {
    NSLog(@"%@", str);
}

最后还有一点 所有的OC方法都有两个默认的参数: 方法的调用者和方法的编号(选择子)后面跟我们平时传入的参数。
在平时写方法的时候一般都省掉了前面两个,因为在运行时,编译器会自动帮我们加上。但是在编写runtime代码的时候要记得加上。不然不会去调用这个方法,而是会沿着继承体系调用父类的方法。

©著作权归作者所有,转载或内容合作请联系作者
平台声明:文章内容(如有图片或视频亦包括在内)由作者上传并发布,文章内容仅代表作者本人观点,简书系信息发布平台,仅提供信息存储服务。

推荐阅读更多精彩内容

  • 转载自:如何自己动手实现 KVO KVO (Key-Value Observing) KVO 是 Objectiv...
    John_LS阅读 4,359评论 0 2
  • 本篇会对KVO的实现进行探究,不涉及太多KVO的使用方法,但是会有一些使用时的思考。 一、使用上的疑问 1.key...
    奋拓达阅读 3,493评论 0 2
  • 转至元数据结尾创建: 董潇伟,最新修改于: 十二月 23, 2016 转至元数据起始第一章:isa和Class一....
    40c0490e5268阅读 5,806评论 0 9
  • 面试题参考1 : 面试题[http://www.cocoachina.com/ios/20150803/12872...
    江河_ios阅读 5,677评论 0 4
  • 过完年,我突然清醒地意识到:二十五岁,已经不是小孩子了,已经进入了慢慢变老了的过程,不能再随意地挥霍青春了。...
    三叶草amy阅读 1,794评论 0 0