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