美文网首页IOS->开发技巧iOS开发新发现iOS知识库
iOS开发 -- KVO的实现原理与具体应用

iOS开发 -- KVO的实现原理与具体应用

作者: 啊左 | 来源:发表于2016-05-03 16:23 被阅读30345次

本文分为2个部分:概念应用
概念部分旨在剖析 KVO 这一设计模式的实现原理;
应用部分通过创建的项目,以说明 KVO 技术在 iOS 开发中所带来的作用;
如果是作为刚接触 KVO 的初学者,可以在了解第一部分的基本原理后粗略看几遍底层实现原理,再认真阅读第二部分的应用内容“学会怎么去使用 KVO,往后再慢慢深入了解 KVO 这一“黑魔法”技术的实现原理。
【本次开发环境:Xcode:7.2 iOS Simulator:iphone6 By:啊左;
本文 Demo下载链接(不忙的话麻烦点个星星😅):KVO演示Demo


[概念部分]

一、KVO 是什么?

KVO 是 Objective-C 对观察者设计模式的一种实现。【另外一种是:通知机制(notification),详情参考:iOS 趣谈设计模式——通知】;
KVO 提供一种机制,指定一个被观察对象(例如 A 类),当对象某个属性(例如 A 中的字符串 name)发生更改时,对象会获得通知,并作出相应处理;【且不需要给被观察的对象添加任何额外代码,就能使用 KVO 机制】

在 MVC 设计架构下的项目,KVO 机制很适合实现 mode 模型和 view 视图之间的通讯。
例如:代码中,在模型类A创建属性数据,在控制器中创建观察者,一旦属性数据发生改变就收到观察者收到通知,通过 KVO 再在控制器使用回调方法处理实现视图 B 的更新;(本文中的应用就是这样的例子.)

二、实现原理?

KVO 在 Apple 中的 API 文档如下:

     Automatic key-value observing is implemented using a technique called 
isa-swizzling… When an observer is registered for an attribute of an object the 
isa pointer of the observed object is modified, pointing to an intermediate class 
rather than at the true class …

KVO 的实现依赖于 Objective-C 强大的 Runtime,从以上 Apple 的文档可以看出苹果对于 KVO 机制的实现是一笔带过,而具体的细节没有过多的描述,但是我们可以通过 Runtime 的所提供的方法去探索【可参考:Runtime的几个小例子】,关于KVO 机制的底层实现原理。为此啊左从网上的一些关于 KVO 的资料总结了有关的内容:

基本的原理:

当观察某对象 A 时,KVO 机制动态创建一个对象A当前类的子类,并为这个新的子类重写了被观察属性 keyPath 的 setter 方法。setter 方法随后负责通知观察对象属性的改变状况。

深入剖析

Apple 使用了 isa 混写(isa-swizzling)来实现 KVO 。当观察对象A时,KVO机制动态创建一个新的名为:NSKVONotifying_A 的新类,该类继承自对象A的本类,且 KVO 为 NSKVONotifying_A 重写观察属性的 setter 方法,setter 方法会负责在调用原 setter 方法之前和之后,通知所有观察对象属性值的更改情况。
(备注: isa 混写(isa-swizzling)isa:is a kind of ; swizzling:混合,搅合;)

①NSKVONotifying_A 类剖析:在这个过程,被观察对象的 isa 指针从指向原来的 A 类,被 KVO 机制修改为指向系统新创建的子类 NSKVONotifying_A 类,来实现当前类属性值改变的监听
所以当我们从应用层面上看来,完全没有意识到有新的类出现,这是系统“隐瞒”了对 KVO 的底层实现过程,让我们误以为还是原来的类。但是此时如果我们创建一个新的名为“NSKVONotifying_A”的类,就会发现系统运行到注册 KVO 的那段代码时程序就崩溃,因为系统在注册监听的时候动态创建了名为 NSKVONotifying_A 的中间类,并指向这个中间类了。
isa 指针的作用:每个对象都有 isa 指针,指向该对象的类,它告诉 Runtime 系统这个对象的类是什么。所以对象注册为观察者时,isa 指针指向新子类,那么这个被观察的对象就神奇地变成新子类的对象(或实例)了。) 因而在该对象上对 setter 的调用就会调用已重写的 setter,从而激活键值通知机制。
—>我猜,这也是 KVO 回调机制,为什么都俗称KVO技术为黑魔法的原因之一吧:内部神秘、外观简洁。
②子类setter方法剖析:KVO 的键值观察通知依赖于 NSObject 的两个方法:willChangeValueForKey:和 didChangevlueForKey:,在存取数值的前后分别调用 2 个方法:
被观察属性发生改变之前,willChangeValueForKey:被调用,通知系统该 keyPath 的属性值即将变更;当改变发生后, didChangeValueForKey: 被调用,通知系统该 keyPath 的属性值已经变更;之后, observeValueForKey:ofObject:change:context: 也会被调用。且重写观察属性的 setter 方法这种继承方式的注入是在运行时而不是编译时实现的。
KVO 为子类的观察者属性重写调用存取方法的工作原理在代码中相当于:

-(void)setName:(NSString *)newName{ 
[self willChangeValueForKey:@"name"];    //KVO 在调用存取方法之前总调用 
[super setValue:newName forKey:@"name"]; //调用父类的存取方法 
[self didChangeValueForKey:@"name"];     //KVO 在调用存取方法之后总调用
}

三、特点:

观察者观察的是属性,只有遵循 KVO 变更属性值的方式才会执行 KVO 的回调方法,例如是否执行了 setter 方法、或者是否使用了 KVC 赋值。
如果赋值没有通过 setter 方法或者 KVC,而是直接修改属性对应的成员变量,例如:仅调用 _name = @"newName",这时是不会触发 KVO 机制,更加不会调用回调方法的。
所以使用 KVO 机制的前提是遵循 KVO 的属性设置方式来变更属性值。


[应用部分]

四、步骤

  • 1.注册观察者,实施监听;
  • 2.在回调方法中处理属性发生的变化;
  • 3.移除观察者

五.实现方法(苹果 API 文档中的方法):

A.注册观察者:

//第一个参数 observer:观察者 (这里观察self.myKVO对象的属性变化)
//第二个参数 keyPath: 被观察的属性名称(这里观察 self.myKVO 中 num 属性值的改变)
//第三个参数 options: 观察属性的新值、旧值等的一些配置(枚举值,可以根据需要设置,例如这里可以使用两项)
//第四个参数 context: 上下文,可以为 KVO 的回调方法传值(例如设定为一个放置数据的字典)
[self.myKVO addObserver:self forKeyPath:@"num" options:
NSKeyValueObservingOptionOld|NSKeyValueObservingOptionNew context:nil]; 

B. 属性(keyPath)的值发生变化时,收到通知,调用以下方法:

//keyPath:属性名称
//object:被观察的对象
//change:变化前后的值都存储在 change 字典中
//context:注册观察者时,context 传过来的值
-(void)observeValueForKeyPath:(NSString *)keyPath ofObject:(id)object change:(NSDictionary<NSString *,id> *)change context:(void *)context
{
}

六、上代码~:

1.新建项目

UI界面设计如下:
第一个是便签,用于显示 num 数值,关联 ViewController 并命名为:label
第二个是按钮,用于改变 num 的数值,关联 ViewController 并命名为:changeNum

2.模型创建

【新建一个 File,选择 Cocoa Touch Class,命名为“myKVO”,记得选择Subclass of “NSObject”.】代码如下:

(myKVO.h):

@interface myKVO : NSObject
@property (nonatomic,assign)int num; //属性设置为int类型的
num@end

(myKVO.m):

#import "myKVO.h"
@implementation myKVO
@synthesize num;
@end

3.在 ViewController 中监听并响应属性改变。

(ViewController.h):

#import <UIKit/UIKit.h>
@interface ViewController : UIViewController
@property (weak, nonatomic) IBOutlet UILabel *label;//便签 label
- (IBAction)changeNum:(UIButton *)sender;           //按钮事件 
@end

(ViewController.m):

#import "ViewController.h"
#import "myKVO.h"
@interface ViewController ()
@property (nonatomic,strong)myKVO *myKVO;
@end

@implementation ViewController

- (void)viewDidLoad { 
[super viewDidLoad]; 

self.myKVO = [[myKVO alloc]init]; 

/*1.注册对象myKVO为被观察者: option中,
NSKeyValueObservingOptionOld 以字典的形式提供 “初始对象数据”; 
NSKeyValueObservingOptionNew 以字典的形式提供 “更新后新的数据”; */ 
[self.myKVO addObserver:self forKeyPath:@"num" options:
NSKeyValueObservingOptionOld|NSKeyValueObservingOptionNew context:nil];
}

/* 2.只要object的keyPath属性发生变化,就会调用此回调方法,进行相应的处理:UI更新:*/
-(void)observeValueForKeyPath:(NSString *)keyPath ofObject:(id)object 
change:(NSDictionary<NSString *,id> *)change context:(void *)context{

// 判断是否为self.myKVO的属性“num”:
if([keyPath isEqualToString:@"num"] && object == self.myKVO) { 
// 响应变化处理:UI更新(label文本改变) 
self.label.text = [NSString stringWithFormat:@"当前的num值为:%@",
[change valueForKey:@"new"]]; 

//change的使用:上文注册时,枚举为2个,因此可以提取change字典中的新、旧值的这两个方法 
NSLog(@"\\noldnum:%@ newnum:%@",[change valueForKey:@"old"],
[change valueForKey:@"new"]); 
}
}

/*KVO以及通知的注销,一般是在-(void)dealloc中编写。
至于很多小伙伴问为什么要在didReceiveMemoryWarning?因为这个例子是在书本上看到的,所以试着使用它的例子。
但小编还是推荐把注销行为放在-(void)dealloc中。(严肃脸😳)
*/
- (void)didReceiveMemoryWarning { 
[super didReceiveMemoryWarning]; 
/* 3.移除KVO */ 
[self.myKVO removeObserver:self forKeyPath:@"num" context:nil];
}

//按钮事件
- (IBAction)changeNum:(UIButton *)sender { 
//按一次,使num的值+1 
self.myKVO.num = self.myKVO.num + 1;
}
@end

调试:便签label初始化没有数值,当每次点击按钮后,label记录的num随之增加,表明按钮使属性num增加的同时,KVO机制发送通知,并调用observeValueForKeyPath:方法使UI更新。(本文Demo下载链接:KVO演示Demo

七、拓展-->

1.KVC 与 KVO 的不同?

KVC(键值编码),即 Key-Value Coding,一个非正式的 Protocol,使用字符串(键)访问一个对象实例变量的机制。而不是通过调用 Setter、Getter 方法等显式的存取方式去访问。
KVO(键值监听),即 Key-Value Observing,它提供一种机制,当指定的对象的属性被修改后,对象就会接受到通知,前提是执行了 setter 方法、或者使用了 KVC 赋值。

2.和 notification(通知)的区别?

notification 比 KVO 多了发送通知的一步。
两者都是一对多,但是对象之间直接的交互,notification 明显得多,需要notificationCenter 来做为中间交互。而 KVO 如我们介绍的,设置观察者->处理属性变化,至于中间通知这一环,则隐秘多了,只留一句“交由系统通知”,具体的可参照以上实现过程的剖析。

notification 的优点是监听不局限于属性的变化,还可以对多种多样的状态变化进行监听,监听范围广,例如键盘、前后台等系统通知的使用也更显灵活方便。
(参照通知机制第五节系统通知名称内容)

3.与 delegate 的不同?

和 delegate 一样,KVO 和 NSNotification 的作用都是类与类之间的通信。但是与 delegate 不同的是:
这两个都是负责发送接收通知,剩下的事情由系统处理,所以不用返回值;而 delegate 则需要通信的对象通过变量(代理)联系;
delegate 一般是一对一,而这两个可以一对多。

4.涉及技术:

KVC/KVO 实现的根本是 Objective-C 的动态性和 Runtime ,以及访问器方法的实现;

总结:

对比其他的回调方式,KVO 机制的运用的实现,更多的由系统支持,相比 notification、delegate 等更简洁些,并且能够提供观察属性的最新值以及原始值;但是相应的在创建子类、重写方法等等方面的内存消耗是很巨大的。所以对于两个类之间的通信,我们可以根据实际开发的环境采用不同的方法,使得开发的项目更加简洁实用。


另外需要注意的是,由于这种继承方式的注入是在运行时而不是编译时实现的,如果给定的实例没有观察者,那么 KVO 不会有任何开销,因为此时根本就没有 KVO 代码存在。但是即使没有观察者,委托和 NSNotification 还是得工作,这也是KVO此处零开销观察的优势。


(转载请标明原文出处,谢谢支持 ~ - ~)
 by:啊左~

相关文章

  • iOS - KVO

    [toc] 参考 KVO KVC 【 iOS--KVO的实现原理与具体应用 】 【 IOS-详解KVO底层实现 】...

  • 整理

    KVO的实现原理与具体应用 2.通知 3.iOS多线程----NSOperation 4.iOS多线程----GC...

  • iOS开发 -- KVO的实现原理与具体应用

    本文分为2个部分:概念与应用。概念部分旨在剖析 KVO 这一设计模式的实现原理;应用部分通过创建的项目,以说明 K...

  • iOS--KVO的实现原理与具体应用(转自看)

    iOS--KVO的实现原理与具体应用 长时间不用容易忘,这篇文章挺好的.转载自看本文分为2个部分:概念与应用。概念...

  • IOS KVO原理解析与应用

    IOS KVO原理解析与应用 一、KVO概述 KVO,即:Key-Value Observing,是Objecti...

  • KVO的实现原理与具体应用

    一、KVO 是什么? KVO 是 Objective-C 对观察者设计模式的一种实现。 KVO 提供一种机制,指定...

  • 2018之开发 -- KVO的实现原理与具体应用

    转载:https://www.jianshu.com/p/e59bb8f59302

  • iOS 自定义KVO

    通过在了解KVO的实现原理和实现步骤之后,我们可以手动实现KVO,具体可以看最后的demo,这里只讲实现原理 添加...

  • KVC

    KVC原理剖析 - CocoaChina_让移动开发更简单 iOS开发底层细究:KVC和KVO底层原理 | iOS...

  • iOS 自定义KVO

    自己实现kvo之前,需要知道iOS系统对kvo的实现。 系统实现kvo的原理 这依赖了OC强大的runtime特性...

网友评论

  • fd5657ed541e:4.运行时能增加成员变量么?能增加属性么?如果能,如何增加?如果不能,为什么?

    这里的回答不妥, objc_setAssociated 方法是把 key-value 存储到全局的HashMap中, 并不会创建是实例变量, 只能提供 属性的setter/getter
  • SSJeremy:willChangeValueForKey,didChangeValueForKey这两个方法的实现有点好奇
    不在dealloc里写是不是理论上有时候系统为了优化程序效率不调用dealloc方法,因为app终止后占用的资源也会返回给系统,不过工程中用大概也没差。
    啊左:@SSJeremy 好像一般都是在 dealloc 中注销的。
    SSJeremy:@啊左 我的意思是者两个方法的实现有点好奇。然后kvo注销不在dealloc里写可能是下面那个原因,不多dealloc里注销也应该没啥问题
    啊左:@SSJeremy 这两个方法为什么要在dealloc?dealloc应该来说是会调用的,像是
    清空对象,如果没有这个来回收对象,不是很容易出现野指针么。
  • MoussyL:谢谢作者的分享,好文~
    But,有个地方测试结果和文中不符,文中说:
    “但是此时如果我们创建一个新的名为“NSKVONotifying_A”的类,就会发现系统运行到注册 KVO 的那段代码时程序就崩溃”
    我在代码里创建了 NSKVONotifying_xxx 类,运行到 addObserve:forKeyPath: 的时候,并没有出现崩溃,但是控制台输出了如下的提示:
    “[general] KVO failed to allocate class pair for name NSKVONotifying_Car, automatic key-value observing will not work for this class”
    观察者失效,不能再观察注册的属性了。
  • 3fdb581ec682:大神写的太好了就是看不懂
    啊左:@南海一页扁舟 :joy:其实已经去掉很多内容,简化很多了。
  • 啊左:by:很多人在问 KVC,下次有空也找下资料写一篇关于 KVC 的,跟大家一起交流。
  • 天蓬大元:1,子类中如果直接调用父类的setter方法赋值,还会不会触发KVO。
    2,既然KVC可以避开setter方法和getter方法,那KVC就不可能通过子类的setter方法触发KVO,那么KVC到底是怎么触发KVO的。KVC和KVO的底层原理到底是什么?(使用KVC改变属性值时,会自动调用willChangeValueForKey和didChangeValueForKey方法,也就是说,KVO的实现其实是这两个方法被调用后的结果?),望作者告知
    啊左:@天蓬大元 后半句听不太明白你说的,但是“KVO会将isa指针由父类转到子类”这个是不会的,因为对象A被KVO纳入观察,是因为新生成了一个“NSKVONotifying_A”子类。
    至于说父类的属性也会触发,情况是A类中有属性num,B是A子类,所以通过“对象B addObserver:self forKeyPath:@"num" ...”注册观察者,那么当num值的改变,也会触发KVO,是因为B继承于A,拥有了A的属性num,你也可以理解为,其实是因为你对B这个类中的num进行了观察。
    天蓬大元:@啊左 第一个问题:你的意思是,改成父类的set方法也会触发KVO是吧。因为,KVO会将isa指针由父类转到子类,注册时,其实已经是子类的对象了。此时中间的这个设置父类属性值的方法只是单纯的赋值。触发KVO的上下的两个方法?可以这么理解吗?
    啊左:1.会不会触发,你得看下这个方法[self.myKVO addObserver:self forKeyPath:@"num" options:
    NSKeyValueObservingOptionOld|NSKeyValueObservingOptionNew context:nil];
    如果说“self.myKVO”改为子类的对象,那么即使属性“num”是父类的,也会触发。
    所以谁会触发取决于哪个类的对象注册了观察者。
    2、使用 KVC 是没有调用 getter 和 setter 方法的,至于如何触发和底层原理,一两句说不清楚,你可以研究下官方的说明:“https://developer.apple.com/library/content/documentation/Cocoa/Conceptual/KeyValueObserving/Articles/KVOCompliance.html#//apple_ref/doc/uid/20002178-BAJEAIEE”,
    或者这篇手动实现KVO的文章:“http://tech.glowing.com/cn/implement-kvo/”,会更清晰一些。
  • 温学振:demo清楚易懂 非常棒
    啊左:谢谢~
  • liyc_dev:- (void)didReceiveMemoryWarning {
    [super didReceiveMemoryWarning];
    /* 3.移除KVO */
    [self removeObserver:self forKeyPath:@"num" context:nil];
    }

    这里移除是用self调用remove吗?
    不是[self.myKVO removeObserver:self forKeyPath:@"num" context:nil];吗?
    啊左:应该是 self.myKVO,哈,回去我看下。
  • 习惯有你syh:@啊左 可以转载吗?注明出处!
    习惯有你syh:@啊左 我也想写一篇关于KVO的,看了一些资料,觉得你写的很好了,我再写也没意思了,等我把你这的研究透了转一下:smile:
    习惯有你syh:@啊左 嗯嗯
    啊左:@习惯有你syh 可以呀,麻烦注明出处。
  • McDan:写得挺好,简洁易懂 不错
  • 421545f21c4d:很棒的文章
    啊左:@车宝大人 谢谢~
  • 知行合一认知升级:@synthesize num;
    现在还用@synthesize 关键字?我是去年才学iOS的,这个关键 字是不是3年前的时候比较流行呀?
    啊左:所以在旧项目上面,也会这样写:
    @synthesize num = _num;
    啊左:也不是说3年前比较流行,现在也会用到。他可以改变属性变量的名称,但是不会影响到@property产生的setter和getter方法的名称。
  • 程序猿马国玺:有人盗你文章
    啊左:@程序猿马国玺 好的,谢谢。
    程序猿马国玺:@啊左 你直接搜一下题目就行了,连题目都没改
    啊左::joy:谢谢,我之前好像看过,麻烦发我下
  • liangdahong:给力:stuck_out_tongue_winking_eye:,KVO为什么必须用子类来实现。如果不用子类有什么方法做到吗?
    啊左:谢谢你的评论,不好意思之前都木有看到你的评论。
    KVO为什么用子类实现,文中以及评论有许多的讨论可以看下。
    至于不用子类有什么方法做到,就很多了,像KVC、通知、delegate都是类似的可在数据改变时通知到外类,只是,如果不用子类的话,哈哈,那就不叫KVO了,这是KVO的一大特性,也是之所以叫它黑魔法的原因吧。:smile:
    啊左: 减少很多意外错误倒不一定,主要是即使产生意外,只会对生成的子类内部产生影响,从而在实现此功能的前提下不会影响到本类。
    34140543b1e5: @idhong 子类没有改变原本的类,因为isa需要指向相同的类才不会改变你原本的想法。第一是方便,不需要重新写一个类。其次继承系统可以拿来就用,至于为什么不用原本的类,我认为是,首先运行期代码不用修改,其次这种类似的copy模式可以减少很多意外错误
  • rockyMJ:总结的很好,很详细.👍
  • XIAODAO:你好,有一个疑问,很多博客都写“KVC是基于isa-swizzling技术”,但是苹果官方文档里我只看到Automatic key-value observing是基于isa-swizzling技术。如果方便的话,请教你一下isa-swizzling技术
    XIAODAO:@啊左 isa-swizzling和Runtime中的Method Swizzling有什么区别吗?
    XIAODAO:@啊左 苹果官方文档KVC部分里并未发现isa-swizzling字样,这也是我困惑的原因。附Apple官方两篇KVC文档:
    1、https://developer.apple.com/library/content/documentation/General/Conceptual/DevPedia-CocoaCore/KeyValueCoding.html
    2、https://developer.apple.com/library/content/documentation/Cocoa/Conceptual/KeyValueCoding/index.html#//apple_ref/doc/uid/10000107-SW1
    啊左:Automatic key-value observing:自动键-值观察,是基于isa-swizzling技术;
    isa指针,其实是包含了指向实现类中的方法的指针,和其它数据。
    而KVC在调用方法setValue的时候,就是通过这种技术来实现其内部查找方法,并实现键值的修改的,有兴趣你可以去多查查或者看看文档,这个内容还是不少的。
  • Cass__:KVO为什么要创建一个子类来实现呢?
    啊左:哈,这个得问苹果了,但是个人觉得应该是在保留原setter 方法的前提下,来实现当前类属性值改变的监听,隐藏内部的实现吧。
  • angolo:写的很好,简洁明了
  • 一剑寒潇:移除KVO观测为什么放在“ didReceiveMemoryWarning”方法中?
    一剑寒潇:@啊左 嗯,应该就是放在dealloc中的
    啊左:@vision123 我想应该是放在dealloc里面,这里举的例子是在《精通iOS开发》书里面的,为了统一所以跟书里一样。🙂
  • iOS打怪升级:你的这些原理阐述是在哪里学的啊,是书上吗,什么书?
    啊左:书上、某些博客的观点、论坛搜的资料、自己试验的demo。
  • 李连毛:找了半天demo,总算找到啦,谢谢啦
    啊左:@李连毛 客气。:grin:
  • 郑明明:本来期待的是在设计模式层面去剖析KVO,但是却是底层代码,不过也不错
    啊左:@NtZheng (ミචᆽචミ)
  • 棍武中原:是的啊
  • 棍武中原:楼主,我在注销监听的时候崩溃,怎么回事removeObserver??就是这步。进入耳机页面,出来在这步崩溃
    啊左:@棍武中原 didReceiveMemoryWarning是在收到内存警告时候调用,你可以把removeObserver改为放在“-(void)dealloc{}”中
  • 就是一个春天的花朵:KVO 的调用方法里面,change这个字典里面的kind是什么啊,怎么每次都是1
    啊左:@ozill 哈,没事,客气。
    就是一个春天的花朵:@啊左 啊啊这样啊,不好意思见笑了,应该想到是一个枚举的,非常感谢!
    啊左:@ozill
    change的kind是个枚举:
    typedef NS_ENUM(NSUInteger, NSKeyValueChange) {
    NSKeyValueChangeSetting = 1,// 设置一个新值。被监听的属性可以是一个对象,也可以是一对一关系的属性或一对多关系的属性。
    NSKeyValueChangeInsertion = 2,// 表示一个对象被插入到一对多关系的属性。
    NSKeyValueChangeRemoval = 3,// 表示一个对象被从一对多关系的属性中移除。
    NSKeyValueChangeReplacement = 4,// 表示一个对象在一对多的关系的属性中被替换
    };
    其实KVO我们主要用的也是New和Old,如果你对Kind感兴趣,也可以结合Xocde的文档,也可以看下刚刚我搜的一篇文章:http://www.bkjia.com/IOSjc/993206.html
  • 郑明明:确实不错的
    郑明明:@啊左 多说两句话不行吗 - -
    啊左:@NtZheng 朋友是来刷评论的么😅
  • 郑明明:2、子类的setter方法到底经历了怎么样的重写
  • 郑明明:讲解了两个点:1、KVO通过runtime机制为被监听对象创建了一个新的类
  • 郑明明:剖析那个地方确实不错
  • 郑明明:确实是剖析,不错的分析,看了之后收获不少呢
  • 一抹淡季:谢谢分享,写的挺赞的👍
  • 字节码:谢谢
  • 字节码:谢谢分享,学习了
  • WhisperKarl:学习了 特别是实现原理
  • 小鲁靠谱:受教了
  • 三只老虎:写的很不错, 32个赞 :clap:
  • 时光的沙漏:简单明了,好文章,加油
    啊左:@Ngltsl 谢谢 :octocat:
  • 时光的沙漏:写得很不错
  • linfantasy:你设置值的时候 并没有用 kvc机制啊
    啊左:@linfantasy 只要遵循 KVO 的属性设置方式,都可以触发KVO机制。本文为了避免跟KVC混淆,故意不使用KVC的。而是使用了执行setter方法改变属性值来触发KVO。你看下第三节的“特点”。
  • emmet7life:写的很好,赞 ~\(≧▽≦)/~
  • ea600a96e7a9:移除 KVO 写在didReceiveMemoryWarning 里面可以生效吗
    奔哥小木屋:为什么不在delloc里面移除呢?
    啊左:@IreliaHua 系统在收到内存警告时会调用didReceiveMemoryWarning,KVO生效。
  • 叫我郎君:很深入,感谢 :+1:
    啊左:@A_L_iu 客气~
  • 做一个有爱的伸手党:你传的demo有问题呀
    啊左:@做一个有爱的伸手党 客气~ :+1: 加油!
    做一个有爱的伸手党:@啊左 谢谢了正在研究这些进阶谢谢你们喽 以后多出好文章
    啊左:@做一个有爱的伸手党 :joy: github客户端上传出问题,已修复,谢谢提醒~
  • d9557f883fd8:delegate其实也可以一对多,用runtime的消息转发
    啊左:@CoderYQ 我没使用过多代理,但是看过相关资料,即使使用 NSPointerArray 可以避免野指针,但是都是提到要用转发,因为:
    这样,复写方法后,可以保证添加在 delegates 数组里面所有能响应这次调用的代理对象都获得调用。
    老板娘来盘一血:多代理不用 runtime消息转发也能实现,比如使用NSPointerArray...
    啊左:@zhnnnnn
    delegate本意是“委托”,要很多人监听的话,代码冗长且复杂,KVO/通知会方便些。
    不过delegate一对多的话,某些场景会有需求,且需要借助runtime。
    在这里阐述他们的不同的时候,主要是强调KVO/通知的一对多的广播本质特性;
    谢谢你的提醒。
  • stormrage曦正:谢了
    啊左:@曦正爱幼 客气~

本文标题:iOS开发 -- KVO的实现原理与具体应用

本文链接:https://www.haomeiwen.com/subject/qrfrrttx.html