[转]NSNotification、delegate和KVO、KVC的区别

1.KVC。

KVC是一种间接访问对象属性的机制,而不是直接通过设置器和访问器或者点语法来访问对象属性。

比如:创建一个学生对象。

Student * student = [[Student alloc] init];

[student setValue@"zhangsan"forKey@"_name"];//通过KVO对student对象的_name变量赋值

NSString * str = [student valueForKey@"_name"];//取出_name的数值,存放在str中

还有一种方法。比如在又定义了一个Teacher类。Teacher类里边同样有一个实例变量_name,在Student类的interface中引入Teacher类,并且创建实例变量Teacher  * _t;

#import <Foundation/Foundation.h>
//#import "Teach.h"
@class Teach;//用这样的方法是为了防止执行到引入的时候发生嵌套,产生死循环。
@interface Student : NSObject
{
    NSString * _name;    
    Teach * _t;
}
@end

Teacherl类的interface。

@interface Teach : NSObject
{
    NSString * _name;
    NSString * _address;
}
@end

在主函数中。

Teacher * teacher  = [[Teacher alloc] init];//创建一个teacher对象。

[student setValue@"teacher"forKey@"_t"];//次数的teacher对象是上边刚刚创建的那个teacher对象

[student setValue@"lisi"ForKeyPath@"_t.name"];//表示路径

NSString * str1 = [student valueForKeyPath@"_t.name"];//通过student来存取teacher中的变量_name.

NSString * str2 = [teacher valueForKey@"_name"];//当输出str2的时候结果和str1是相同的都是“lisi”。

2.KVO。

当对象的某一个属性发生变化的时候,我们得到一个相应的通知。

主函数中: 

Teach * teacher = [[Teach alloc]init];
    teacher.name = @"abc";//在声明文件中声明了name属性。所以直接使用点语法赋值
    [teacher addObserver:teacher   //teacher为事件的监听者,此处就是注意谁监听谁实现后边的 监听触发以后的方法。
              forKeyPath:@"name" //便是监听的哪一个属性
              options:NSKeyValueObservingOptionNew |NSKeyValueObservingOptionOld//得到什么值
              context:nil];
    teacher.name = @"123";//一旦name发生变换,就进入到teacher的实现文件(.m)中调用方法
    teacher.name = @"456";//每次都会调用那个方法,,实时更新

在Teacher类的.m文件中

#import "Teach.h"
@implementation Teach
@synthesize  name = _name;
 -(void)observeValueForKeyPath:(NSString *)keyPath ofObject:(id)object change:(NSDictionary *)change context:(void *)context
{
    NSLog(@"keyPath:%@, object:%@  change:%@  context:%@",keyPath,object,change,context);
}
@end
打印结果:

2012-08-13 08:48:34.162 KVC[422:14503] keyPath:name, object:<Teach: 0x8a3ee10>  change:{
    kind = 1;
    new = 123;  
    old = abc;
}  context:(null)
2012-08-13 08:48:34.164 KVC[422:14503] keyPath:name, object:<Teach: 0x8a3ee10>  change:{
    kind = 1;
    new = 456;
    old = 123;
}  context:(null)

3.NSNotification 通知

NSNotification的对象很简单,就是poster要提供给observer的信息包裹。

在主函数中创建监听通知的过程。

NSNotificationCenter * center = [NSNotificationCenter defaultCenter];//通过单例得到通知中心,相当于通知在什么地方发布(黑板)

Teach * teacher = [[Teach alloc]init];

[center addObserver: teacher  selector:@selector(test:) name:@"黑苹果" object:nil];//self是监听者,selector是表示监听到一个通知以后需要执行里边的test:方法。name表示需要监听的通知是什么内容。object表示监听通知的发出者。

NSNotification * notification1 = [NSNotification notificationWithName:@"黑苹果" object:teacher  userInfo:nil];创建一个通知

[center postNotification:notification]//通过上边建立的通知中心center发布通知notification1.

[center postNotificationWithName:@"黑苹果" object:teacher  userInfo:nil]//作用相当于上边2行代码。

在监听者teacher的.m文件加加入test:方法

-(void)test:(NSNotification *)n
{
    NSLog(@"name : %@  boject = %@  userInfo = %@",[n name],[n object],[n userInfo]);
}
打印结果:

2012-08-13 09:13:35.328 KVC[452:14503] name : 黑苹果  boject = <Teach: 0x6b2b350>  userInfo = (null)

 

 

1.效率肯定是delegate比nsnotification高。

2. delegate方法比notification更加直接,最典型的特征是,delegate方法往往需要关注返回值, 也就是delegate方法的结果。比如-windowShouldClose:,需要关心返回的是yes还是no。所以delegate方法往往包含 should这个很传神的词。也就是好比你做我的delegate,我会问你我想关闭窗口你愿意吗?你需要给我一个答案,我根据你的答案来决定如何做下一 步。相反的,notification最大的特色就是不关心接受者的态度, 我只管把通告放出来,你接受不接受就是你的事情,同时我也不关心结果。所以notification往往用did这个词汇,比如 NSWindowDidResizeNotification,那么nswindow对象放出这个notification后就什么都不管了也不会等待接 受者的反应。

简明概要的说明了KVO和NSNotification的区别:

和delegate一样,KVO和NSNotification的作用也是类与类之间的通信,与delegate不同的是1)这两个都是负责发出通知,剩下的事情就不管了,所以没有返回值;2)delegate只是一对一,而这两个可以一对多。这两者也有各自的特点。

1)KVO的使用:

被观察者发出  addObserver:forKeyPath:options:context:  方法来添加观察者。

然后只要被观察者的keyPath值变化(注意:单纯改变其值不会调用此方法,只有通过getters和setters来改变值才会触发KVO),就会在观察者里调用方法observeValueForKeyPath:ofObject:change:context:

因此观察者需要实现方法 observeValueForKeyPath:ofObject:change:context: 来对KVO发出的通知做出响应。

这 些代码都只需在观察者里进行实现,被观察者不用添加任何代码,所以谁要监听谁注册,然后对响应进行处理即可,使得观察者与被观察者完全解耦,运用很灵活很 简便;但是KVO只能检测类中的属性,并且属性名都是通过NSString来查找,编译器不会帮你检错和补全,纯手敲所以比较容易出错。
2)NSNotification的使用
这里的通知不是由被观察者发出,而是由NSNotificationCenter来统一发出,而不同通知通过唯一的通知标识名notificationName来区分,标识名由发送通知的类来起。
首先被观察者自己在必要的方法A里,通过方法postNotificationName:object:来发出通知notificationName这样发送通知者这边的工作就完成了,每次A被调用,就会发送一次通知notificationName。
然后谁要监听A的变化,就通过[NSNotificationCenter defaultCenter]的方法addObserver:selector:name:object:为观察者注册监听name为notificationName的通知然后每次发出name为notificationName的通知时,注册监听后的观察者就会调用其自己定义的方法notificationSelector来进行响应。
NSNotification的特点呢,就是需要被观察者先主动发出通知,然后观察者注册监听后再来进行响应,比KVO多了发送通知的一步,但是其优点是监听不局限于属性的变化,还可以对多种多样的状态变化进行监听,监听范围广,使用也更灵活。
posted @ 2016-09-12 11:06  超级大熊  阅读(179)  评论(0编辑  收藏  举报