iOS-底层原理21-KVO(下)
《iOS底层原理文章汇总》
上一篇文章《iOS-底层原理20-KVO(上)》介绍到自定义KVO中观察到属性的值发生变化后,怎么通知到自定义的方法中
1.自定义KVO,属性值变化后通知到自定义方法中
来到lg_setter方法中之后,改变父类中nickName的值,进行消息发送,往父类发送消息setNickName:,重定义objc_msgSendSuper,传入三个参数objc_super父类结构体指针,_cmd(setNickName:)方法,newValue新改变的值,进入自定义子类LGKVONotifying_LGPerson的父类LGPerson的setNickName:方法中
// 4: 消息转发 : 转发给父类
// 改变父类的值 --- 可以强制类型转换
void (*lg_msgSendSuper)(void *,SEL , id) = (void *)objc_msgSendSuper;
// void /* struct objc_super *super, SEL op, ... */
struct objc_super superStruct = {
.receiver = self,
.super_class = class_getSuperclass(object_getClass(self)),
};
//objc_msgSendSuper(&superStruct,_cmd,newValue)
lg_msgSendSuper(&superStruct,_cmd,newValue);
data:image/s3,"s3://crabby-images/4743f/4743f0f3fddf53075d2b6038ca41a1a80adbf2ba" alt=""
data:image/s3,"s3://crabby-images/adac1/adac197379b39ad3d58aa833525a5cd6f90324d2" alt=""
-
将llvm严格校验参数个数关闭,解决objc_msgSendSuper(struct objc_super * _Nonnull super, SEL _Nonnull op, ...)方法报错
image.png
-
此时存在一个问题,系统的观察者在观察前后一直self.person.class一致指向LGPerson,而此时自定义的观察者self.person.class指向LGKVONotifying_LGPerson,此时需要重写class方法
image.png
image.png
重写class方法:class_getSuperclass(object_getClass(self))
image.png
Class lg_class(id self,SEL _cmd){
return class_getSuperclass(object_getClass(self));
}
data:image/s3,"s3://crabby-images/fd660/fd660ce4576cdcd2d6d620041e39891664b17530" alt=""
下面这种写法会死循环:class_getSuperclass([self class])中[self class]还会调用lg_class方法
data:image/s3,"s3://crabby-images/3a6a7/3a6a7fdf5872fe9835c93117ae822f254fa79599" alt=""
data:image/s3,"s3://crabby-images/c8d9f/c8d9f0c2e110e7150f854a0d095cb02d97169df1" alt=""
此时self.person.class的指向才和系统观察者一模一样了
data:image/s3,"s3://crabby-images/d6056/d6056d8f97b5dd5675626c0820069e9eef537bb0" alt=""
此时执行程序发现无法找到setNickName:方法,为什么呢?分析原因,取了两次父类.super_class = class_getSuperclass([self class])中[self class]调用lg_class方法,class_getSuperclass(object_getClass(self))又取一次父类,即LGKVONotifying_LGPerson的父类的父类,并不存在setNickName:所以报unrecognized selector
data:image/s3,"s3://crabby-images/a5fe3/a5fe351e7415305c983fd5fa0e99d76a8f49cb30" alt=""
data:image/s3,"s3://crabby-images/8edc6/8edc6eb05c2f731c06cb976aa6c909c0edea4d0a" alt=""
正确的写法为.super_class = [self class]LGKVONotifying_LGPerson的父类LGPerson中查找setNickName:方法
data:image/s3,"s3://crabby-images/d60f8/d60f831635d6e01d1d796ea747ae684b4a5eb9a7" alt=""
- objc_msgSendSuper消息发送的好处:通用,封装,解决了依赖,并不依赖于类LGPerson或NSObject,都不用关注这个类是NSObject还是LGPerson
void (*lg_msgSendSuper)(void *,SEL , id) = (void *)objc_msgSendSuper;
// void /* struct objc_super *super, SEL op, ... */
struct objc_super superStruct = {
.receiver = self,
.super_class = [self class],
};
//objc_msgSendSuper(&superStruct,_cmd,newValue)
lg_msgSendSuper(&superStruct,_cmd,newValue);
data:image/s3,"s3://crabby-images/3e25a/3e25a7e6cbb630e020e7ad7cdb2048497bc2831c" alt=""
-
通知VC中的方法,属性值发生了变化,先要保存VC,在分类中保存VC,用到runtime关联对象,保存观察者VC控制器,setNickName:中收到nickName的值变化后,发送消息通知观察者控制器VC
image.png
image.png
-
添加关联对象是否产生循环引用了?没有,因为关联对象是存放在一个哈希表中,并没有进行持有,一一对应关系的保存,控制器VCpop后还是能正常进入dealloc方法,并没有产生循环引用
AssociationsHashMap@2x.png
image.png
image.png
2.若要观察多个属性呢,以上会保存多次观察者VC,这里需要优化。
-
1.新建LGKVOInfo保存观察信息类,保存观察者信息
image.png
image.png
-
2.观察一个属性的情况
I.添加观察者,观察属性nickName变化
image.png
II.保存观察者信息,方便第三步进行消息发送到观察者控制器
image.png
III.对新旧值进行处理,发送给观察者
image.png
-
3.观察多个属性值的情况,同事观察nickName和kcName
image.png
image.png
image.png
观察两个属性,observerArr中存在两个info
image.png
3.移除观察者
data:image/s3,"s3://crabby-images/9505c/9505c01731b15e222025b43c058042c7d0d74c11" alt=""
以上自定义的KVO是通过传值的方法,观察到值的变化后,在不同的方法中进行通知,有没有函数式的方式y=f(x)的方式,和添加观察者耦合再一起呢?
4.自定义函数式KVO
data:image/s3,"s3://crabby-images/7ffb2/7ffb2569ce3bec6afd8b18ac762ce7266df357e3" alt=""
data:image/s3,"s3://crabby-images/9ffb9/9ffb9f3d95da2fcb0d1bfd5513f00d4f0e89d3c6" alt=""
data:image/s3,"s3://crabby-images/6209d/6209dc2fa1adca292969a4c8ea41d0a6a78959e8" alt=""
data:image/s3,"s3://crabby-images/febfd/febfd276e2936037cd993adb46ad8d0b4176aa89" alt=""
data:image/s3,"s3://crabby-images/74e28/74e284aa33f892c977e1d2e5d40e793ff0b948c2" alt=""
5.KVO自动销毁机制
在VC的dealloc方法中调用移除观察者方法[self.person lg_removeObserver:self forKeyPath:@"nickName"]
,怎么对观察者进行自动移除呢?在什么时候自动移除呢?VC进行销毁的时候,会调用dealloc方法,VC销毁,则持有的属性LGPerson必定销毁了,在VC的dealloc()方法中会给LGPerson发送release消息[self.person release]
,我们可以监听self.person是什么时候销毁的,即观察LGPerson是什么时候析构(走入-(void)dealloc方法)的,要想将自定义观察者自动移除,我们想到一种办法是监听LGPerson的析构函数-(void)dealloc,并在析构函数中将LGPerson的isa由LGKVONotifying_LGPerson指回LGPerson
data:image/s3,"s3://crabby-images/ee151/ee151ada33432b5c3f1beed1916983ab3b7403e8" alt=""
我们不能在NSObject分类中直接重写-(void)dealloc方法进行监听,会改变所有NSObject子类的-(void)dealloc方法,可以在添加观察者的时候进行方法交换从而改变LGPerson的isa指向,此刻会循环递归调用,程序崩溃
data:image/s3,"s3://crabby-images/a681c/a681cb7d5ad6660bb076a0090d640e3971ee4e7e" alt=""
循环递归调用的原因是LGPerson中没有实现dealloc方法,则会取LGPerson的父类NSObject也就是系统中的-(void)dealloc方法进行交换,NSObject中的dealloc方法就被替换为了NSObject+LGKVO分类中myDealloc方法的实现,-(void)myDealloc方法被替换为了NSObject中的dealloc方法的实现,但是其他NSObject的子类并不知道系统的dealloc方法已经被替换为myDealloc方法了,所以程序会在走[self myDealloc]中一直循环调用,自己调用自己。
data:image/s3,"s3://crabby-images/8e061/8e061d6e40320ed721438001b41e19d34d3c2bab" alt=""
data:image/s3,"s3://crabby-images/a2e64/a2e64e00bdf18e5484dca6e676b69ead1b190ac4" alt=""
解决办法在LGPerson中实现要被交换的-(void)dealloc方法,程序不会崩溃,LGViewController控制器pop后,里面的属性LGPerson也正常销毁,进入-(void)dealloc方法,可以通过交换方法监听-(void)dealloc方法从而监听到VC的销毁,从而移除自定义观察者,但是又会有别的问题,方法交换的方式就不怎么好,有太多问题了,容易造成系统的混乱
data:image/s3,"s3://crabby-images/a673a/a673ac6ac034c15e4b2dd677dd921b0d02acdf94" alt=""
data:image/s3,"s3://crabby-images/d13f4/d13f471acb5025082888e5cd469bfbafdd8a3cc5" alt=""
由前文知道KVO会生成派生子类LGKVONotifying_LGPerson,重写四个方法class ,setter,dealloc,isKVO,因此当观察者被移除的时候,可以在dealloc方法中将LGPerson对象中的isa指回LGPerson,在VC销毁时进入vc的dealloc方法,后进入LGKVONotifying_LGPerson的dealloc也就是lg_dealloc方法来将isa指回LGPerson
data:image/s3,"s3://crabby-images/b9901/b9901b10cc31cf9248ffc326bc9832d6708e9d31" alt=""
6.FBKVOController源码探索
1.保存信息_FBKVOInfo中用weak修饰下FBKVOController为了防止强引用,循环引用链条为self -> kvoCtrl -> _objectInfosMap -> infos - > info -/弱引用__weak-> self.kvoCtrl
data:image/s3,"s3://crabby-images/3a346/3a3467f4b5fe0bcaa41c29fa7b7ca31c602c56e9" alt=""
data:image/s3,"s3://crabby-images/97e96/97e968b16b307aa122937def53b40309f6d42816" alt=""
2.监听原理
data:image/s3,"s3://crabby-images/c56ac/c56acee8dfb3b50cf306306689ebf7ba5535f6fc" alt=""
7.GNU源码分析:系统的KVO的底层代码就在此
data:image/s3,"s3://crabby-images/6c613/6c61339531e25eb0b8035af021f62c9b1de407b6" alt=""
重写setter方法[r overrideSetterFor: aPath]
添加观察者
[info addObserver: anObserver
forKeyPath: aPath
options: options
context: aContext];
data:image/s3,"s3://crabby-images/2f61a/2f61aab399b31ede5cc393362568abafc10fe19a" alt=""
观察的值发生改变走setter方法,消息发送给父类,判断自动开关是否打开
data:image/s3,"s3://crabby-images/af5e2/af5e2e932aee5e9567f0ef6ccdb4dfa40eb207a6" alt=""
发送响应通知在didChangeValueForKey方法中,对change进行处理,拿到info信息进行通知
data:image/s3,"s3://crabby-images/c7be3/c7be314b5766b277a0dc2df1765b09c7bf5b2793" alt=""
change进行setObject,oldValue和newValue进行处理,将所有消息处理完,向外界发送消息,使外界都能接收到消息,oldValue和newValue分别进行release,整个过程更加符合苹果写的系统的原生集的KVO的处理。
data:image/s3,"s3://crabby-images/0add1/0add12053b7ed5e4756db953bd23113de15ebc1c" alt=""
GNU源码有助于理解Foudation框架
问题一:LGPerson中实现-(void)dealloc方法后,和自定义添加的-(void)lg_setter方法,在LGPerson销毁的时候会走哪一个方法呢???
前文运行发现会走-(void)lg_setter方法,为什么呢?
self.person对象的地址不会发生变化,添加观察者后只是产生了一个动态子类,只是isa改变的一个类型,外部的实例对象地址是不变的,对原来的类LGPerson的-(void)dealloc方法进行了重写和覆盖,并不会再进入LGPerson的-(void)dealloc方法了
data:image/s3,"s3://crabby-images/b7e34/b7e34e4ae036b4d66a740bcd4cae51a0441efb4b" alt=""
问题二:FBKVOController在进行移除观察者的时候,新建的_FBKVOInfo *info是怎么准确的移除的?很明显临时变量info在infos中是找不到的,registeredInfo为nil,所以最后[[_FBKVOSharedController sharedController] unobserve:object info:registeredInfo]将移除个锤子,添加时是临时创建的info,移除时也是临时创建的info,是不同的一个
data:image/s3,"s3://crabby-images/d83e5/d83e5799920714af0c71503611250bc243c91e64" alt=""
data:image/s3,"s3://crabby-images/4d924/4d92449b2180a0e0c4acd027729f73299382c854" alt=""
例如两个person是不同的内存地址,就不相等,info为空,查看NSSet中方法member:官方源码,对象被认为相等,如果isEqual:中的条件是相等的返回YES,则认为对象是相等的
data:image/s3,"s3://crabby-images/44707/44707f6d0be56e280d90de739f367d4dbe27d9bc" alt=""
data:image/s3,"s3://crabby-images/b5005/b5005142e9292fe49650a2f2f18a7d127509f57a" alt=""
重写hash和isEqual:方法来判断对象是否相等,hash匹配的是地址指针的值,两个对象相等哈希值也要相等
data:image/s3,"s3://crabby-images/d7bf2/d7bf2c24c5cad7af7ec3010c4c0e24a9d2ca3055" alt=""
data:image/s3,"s3://crabby-images/b0a93/b0a93a1d5a11d8abac90412102296ca8c92ec7b6" alt=""
data:image/s3,"s3://crabby-images/a2d41/a2d41680f84179f7307b1ccb0c09a5eb007e29f9" alt=""
_FBKVOInfo中有重写hash和isEqual:方法,新建的临时变量的keyPath是相等的,能从_objectInfosMap的infos中找到info,从而进行移除
data:image/s3,"s3://crabby-images/b5b7b/b5b7b7d61469871a6f51d261f82eb672a3ab2232" alt=""
网友评论