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 *)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代码的时候要记得加上。不然不会去调用这个方法,而是会沿着继承体系调用父类的方法。

你可能感兴趣的:(runtime 实现 kvo)