KVC、KVO、Notification、Delegate代码实现及比较

一、KVC 即是指 NSKeyValueCoding,一个非正式的Protocol,提供一种机制来间接访问对象的属性。而不是通过调用Setter、Getter方法访问。KVO 就是基于 KVC 实现的关键技术之一。

?


1

2

3

4

5

6

7

8

9

10

11

12

13

14

15

16

17

18

19

20

21

22

23

24

25

26

27

28

29

30

31

32

33

34

35

36

37

38

39

40

41

42

43

44

45

  

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

        //通过KVC设置name的值

        [student setValue:@"Jacedy" forKey:@"name"];    //等效于:student.name = @"Jacedy"

        NSString *m_name = [student valueForKey:@"name"];

        NSLog(@"%@", m_name);

         

        Course *course = [[Course alloc] init];

        [course setValue:@"音乐" forKey:@"CourseName"];

        [student setValue:course forKey:@"course"];

        //通过键值径获取CourseName的值(KVC按照键值路径取值时,如果对象不包含指定的键值,会自动进入对象内部,查找对象属性)

        NSString *courseName = [student valueForKeyPath:@"course.CourseName"];

        NSLog(@"课程名称:%@", courseName);

        //通过键值径设置CourseName的值

        [student setValue:@"实验课" forKeyPath:@"course.CourseName"];

        courseName = [student valueForKeyPath:@"course.CourseName"];

        NSLog(@"课程名称:%@", courseName);

         

        //通过KVC设置NSInteger的值(使用KVC间接修改对象属性时,系统会自动判断对象属性的类型,并完成转换)

        [student setValue:@"88" forKeyPath:@"point"];

        NSString *m_point = [student valueForKey:@"point"];

        NSLog(@"分数:%@", m_point);

         

        //通过KVC操作集合

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

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

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

        [student1 setValue:@"65" forKey:@"point"];

        [student2 setValue:@"77" forKey:@"point"];

        [student3 setValue:@"99" forKey:@"point"];

        NSArray *array = [NSArray arrayWithObjects:student1, student2, student3, nil];

        [student setValue:array forKey:@"otherStudent"];

        NSLog(@"其他学生的成绩:%@", [student valueForKeyPath:@"otherStudent.point"]);

        //KVC的简单运算

        NSLog(@"共 %@ 个学生", [student valueForKeyPath:@"otherStudent.@count"]);

        NSLog(@"最高成绩:%@", [student valueForKeyPath:@"otherStudent.@max.point"]);

        NSLog(@"最低成绩:%@", [student valueForKeyPath:@"otherStudent.@min.point"]);

        NSLog(@"平均成绩:%@", [student valueForKeyPath:@"otherStudent.@avg.point"]);

         

        /* KVC需要注意的地方:

         1)key的值必须正确,如果拼写错误,会出现异常;

         2)当key的值没有定义时,valueForUndefinedKey:方法会被调用,如果你自己写了这个方法,key的值出错就会调用到这里来;

         3)因为类key反复嵌套,所以有个keyPath的概念,keyPath就是用点.号来把一个一个key链接起来,这样就可以根据这个路径访问下去;

         4)NSArray、NSSet等都支持KVC

        */

二、KVO 的是KeyValue Observe的缩写,中文是键值观察。这是一个典型的观察者模式,观察者在键值改变时会得到通知。iOS中有个Notification的机制,也可以获得通知,但这个机制需要有个Center,相比之下KVO更加简洁而直接。

使用步骤:

1.注册需要观察的对象的属性addObserver:forKeyPath:options:context:
2.实现observeValueForKeyPath:ofObject:change:context:方法,这个方法当观察的属性变化时会自动调用
3.取消注册观察removeObserver:forKeyPath:context:

?


1

2

3

4

5

6

7

8

9

//  JKChild.h

 

#import <Foundation/Foundation.h>

 

@interface JKChild : NSObject

 

@property(nonatomic, assign) NSInteger happyVal;

 

@end

?


1

2

3

4

5

6

7

8

9

10

11

12

13

14

15

16

17

18

19

20

21

22

23

24

25

//  JKChild.m

 

#import "JKChild.h"

 

@implementation JKChild

 

- init {

    if (self = [super init]) {

        self.happyVal = 100;

        //定时器,1秒钟调用一次timerAction:函数

        [NSTimer scheduledTimerWithTimeInterval:1 target:self selector:@selector(timerAction:) userInfo:nil repeats:YES];

    }

    return self;

}

 

- (void)timerAction:(NSTimer *)timer {

    //方式一:

    self.happyVal--;

     

    //方式二:

//    _happyVal--;      //直接修改不会触发监听,还需通过KVC方式设置

//    [self setValue:[NSNumber numberWithInteger:_happyVal] forKey:@"happyVal"];

}

 

@end

?


1

2

3

4

5

6

7

8

9

10

11

12

13

//  JKNurse.h

 

#import <Foundation/Foundation.h>

 

@class JKChild;

 

@interface JKNurse : NSObject

 

@property(nonatomic, retain) JKChild *child;

 

- (id)initWithChild:(JKChild *)child;

 

@end

?


1

2

3

4

5

6

7

8

9

10

11

12

13

14

15

16

17

18

19

20

21

22

23

24

25

26

27

28

29

30

31

32

33

//  JKNurse.m

 

#import "JKNurse.h"

#import "JKChild.h"

 

@implementation JKNurse

 

- (id)initWithChild:(JKChild *)child {

    if (self = [super init]) {

         

        self.child = [child retain];

         

        //KVO 注册监听,监听JKChild类中happyVal的值变化

        [self.child addObserver:self forKeyPath:@"happyVal" options:NSKeyValueObservingOptionNew | NSKeyValueObservingOptionOld context:@"xxx"];

    }

    return self;

}

 

// 监听响应

- (void)observeValueForKeyPath:(NSString *)keyPath ofObject:(id)object change:(NSDictionary<NSString *,id> *)change context:(void *)context {

     

    NSLog(@"keyPath:%@, object:%@,change:%@, context:%@", object, keyPath, change, context);

}

 

- (void)dealloc {

    // 移除监听

    [self.child removeObserver:self forKeyPath:@"happyVal"];

     

    [self.child release];

    [super dealloc];

}

 

@end

?


1

2

3

4

5

6

7

8

9

10

11

12

13

14

15

16

17

//  main.m

 

#import <Foundation/Foundation.h>

#import "JKChild.h"

#import "JKNurse.h"

 

int main(int argc, const char * argv[]) {

    @autoreleasepool {

         

        JKChild *child = [[JKChild alloc] init];

        JKNurse *nurse = [[JKNurse alloc] initWithChild:child];

         

        //加入了定时器,通过runloop使事件持续运行

        [[NSRunLoop currentRunLoop] run];

    }

    return 0;

}

三、Notification(通知)

?


1

2

3

4

5

6

7

8

9

10

11

//  Child.h

 

#import <Foundation/Foundation.h>

 

#define WEEK_INFOMATION @"WEEK"

 

@interface Child : NSObject

 

@property (nonatomic, assign) NSInteger sleep;

 

@end

?


1

2

3

4

5

6

7

8

9

10

11

12

13

14

15

16

17

18

19

20

21

22

23

24

25

26

27

28

29

30

31

32

33

//  Child.m

 

#import "Child.h"

 

@implementation Child

 

- (instancetype) init

{

    self = [super init];

    if (self != nil) {

        _sleep = 100;

         

        // 添加定时器

        [NSTimer scheduledTimerWithTimeInterval:1 target:self selector:@selector(timerAction:) userInfo:nil repeats:YES];

    }

    return self;

}

 

- (void)timerAction:(NSTimer *)timer

{

    _sleep -= 4;

    NSLog(@"%ld", (long)_sleep);

     

    if (_sleep < 90) {

        // 获取通知中心的单例后,给指定的名称发送通知

        [[NSNotificationCenter defaultCenter] postNotificationName:WEEK_INFOMATION object:[NSNumber numberWithInteger:_sleep]];

         

        // 停止定时器

        [timer invalidate];

    }

}

 

@end

?


1

2

3

4

5

6

7

//  Father.h

 

#import <Foundation/Foundation.h>

 

@interface Father : NSObject

 

@end

?


1

2

3

4

5

6

7

8

9

10

11

12

13

14

15

16

17

18

19

20

21

22

23

24

25

//  Father.m

 

#import "Father.h"

#import "Child.h"

 

@implementation Father

 

- (instancetype)init

{

    self = [super init];

    if (self != nil) {

        // 监听接收通知

        [[NSNotificationCenter defaultCenter] addObserver:self selector:@selector(weekNotification:) name:WEEK_INFOMATION object:nil];

    }

    return self;

}

 

- (void)weekNotification:(NSNotification *)notification

{

    NSLog(@"Father received object is : %@", notification.object);

     

    NSLog(@"week up!");

}

 

@end

Mother类代码与Father类代码相似,此处略过......

四、Delegate(代理)

?


1

2

3

4

5

6

7

8

9

10

11

12

13

14

//  Boss.h

 

#import <Foundation/Foundation.h>

#import "Sec.h"

 

@interface Boss : NSObject          //老板类

 

@property(strong, nonatomic) NSString *name;

 

@property(strong, nonatomic) Sec<SecDelegate> *delegate;

 

-(void)work;

 

@end

?


1

2

3

4

5

6

7

8

9

10

11

12

13

14

//  Boss.m

 

#import "Boss.h"

 

@implementation Boss

 

@synthesize name;

 

-(void)work

{

    NSLog(@"%@ 正在工作", name);

}

 

@end

?


1

2

3

4

5

6

7

8

9

10

//  Sec.h

 

#import <Foundation/Foundation.h>

#import "SecDelegate.h"

 

@interface Sec : NSObject <SecDelegate>         //秘书类

 

@property(strong, nonatomic) NSString *name;

 

@end

?


1

2

3

4

5

6

7

8

9

10

11

12

13

14

//  Sec.m

 

#import "Sec.h"

 

@implementation Sec

 

@synthesize name;

 

-(void)phone

{

    NSLog(@"%@ 接到了电话", name);

}

 

@end

?


1

2

3

4

5

6

7

8

9

10

11

12

13

14

15

//  SecDelegate.h

 

#ifndef SecDelegate_h

#define SecDelegate_h

 

#import <Foundation/Foundation.h>

 

@protocol SecDelegate <NSObject>

 

@optional    // 默认为@required

-(void)phone;   // 接电话

 

@end

 

#endif /* SecDelegate_h */

?


1

2

3

4

5

6

7

8

9

10

11

12

13

14

15

16

17

18

19

20

21

22

23

//  main.m

//  代理模式

 

#import <Foundation/Foundation.h>

#import "Boss.h"

#import "Sec.h"

 

int main(int argc, const char * argv[]) {

    @autoreleasepool {

         

        Boss *boss = [[Boss alloc] init];

        [boss setName:@"刘老板"];

         

        Sec *sec = [[Sec alloc] init];

        [sec setName:@"张秘书"];

         

        boss.delegate = sec;

         

        [boss work];

        [sec phone];

    }

    return 0;

}

比较

1)delegate 的 优势 : 

     1.非常严格的语法。所有将听到的事件必须是在delegate协议中有清晰的定义。

     2.如果delegate中的一个方法没有实现那么就会出现编译警告/错误

     3.协议必须在controller的作用域范围内定义

      4.在一个应用中的控制流程是可跟踪的并且是可识别的;

     5.在一个控制器中可以定义定义多个不同的协议,每个协议有不同的delegates

     6.没有第三方对象要求保持/监视通信过程。

     7.能够接收调用的协议方法的返回值。这意味着delegate能够提供反馈信息给controller

      缺点 : 

     1.需要定义很多代码:1.协议定义;2.controller的delegate属性;3.在delegate本身中实现delegate方法定义

     2.在释放代理对象时,需要小心的将delegate改为nil。一旦设定失败,那么调用释放对象的方法将会出现内存crash

     3.在一个controller中有多个delegate对象,并且delegate是遵守同一个协议,但还是很难告诉多个对象同一个事件,不过有可能。

2)notification的 优势 :

       1.不需要编写多少代码,实现比较简单;

       2.对于一个发出的通知,多个对象能够做出反应,即1对多的方式实现简单

       3.controller能够传递context对象(dictionary),context对象携带了关于发送通知的自定义的信息

       缺点 : 

       1.在编译期不会检查通知是否能够被观察者正确的处理; 

       2.在释放注册的对象时,需要在通知中心取消注册;

       3.在调试的时候应用的工作以及控制过程难跟踪;

       4.需要第三方对喜爱那个来管理controller与观察者对象之间的联系;

       5.controller和观察者需要提前知道通知名称、UserInfodictionary keys。如果这些没有在工作区间定义,那么会出现不同步的情况;

       6.通知发出后,controller不能从观察者获得任何的反馈信息。

3)KVO的 优势 :

        1.能够提供一种简单的方法实现两个对象间的同步。例如:model和view之间同步;

        2.能够对非我们创建的对象,即内部对象的状态改变作出响应,而且不需要改变内部对象(SKD对象)的实现;

        3.能够提供观察的属性的最新值以及先前值;

        4.用key paths来观察属性,因此也可以观察嵌套对象;

        5.完成了对观察对象的抽象,因为不需要额外的代码来允许观察值能够被观察

       缺点 :

        1.我们观察的属性必须使用strings来定义。因此在编译器不会出现警告以及检查;

        2.对属性重构将导致我们的观察代码不再可用;

        3.复杂的“IF”语句要求对象正在观察多个值。这是因为所有的观察代码通过一个方法来指向;

        4.当释放观察者时不需要移除观察者。

4) 效率肯定是delegate比NSNotification高。

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

5)KVO和NSNotification的区别:

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

6)delegate针对one-to-one关系,并且reciever可以返回值给sender;notification 可以针对one-to-one/many/none,reciever无法返回值给sender;所以,delegate用于sender希望接受到reciever的某个功能反馈值,notification用于通知多个object某个事件。

时间: 2024-07-31 11:32:08

KVC、KVO、Notification、Delegate代码实现及比较的相关文章

KVC/KVO原理详解及编程指南(转)

作者:wangzz 原文地址:KVC/KVO原理详解及编程指南 本文只转载KVC部分讲解,KVO部分可到原博客查看. 本人在阅读过程中也进行了一些勘误修改. 前言 1.本文基本不讲KVC/KVO的用法,只结合网上的资料说说对这种技术的理解. 2.由于KVO内容较少,而且是以KVC为基础实现的,本文将着重介绍KVC部分. 一.简介 KVC/KVO是观察者模式的一种实现,在Cocoa中是以被万物之源NSObject类实现的NSKeyValueCoding/NSKeyValueObserving非正式

KVC &amp; KVO

        KVC和KVO看上去又是两个挺牛的单词简写,KVC是Key-Value Coding的简写,是键值编码的意思.KVO是Key-Value  Observing的简写,是键值观察的意思.那么我们能拿KVC和KVO干些什么事呢?这两个缩写单词不能否认听起来挺高端的样子.这两个方法都是runtime方法,我们先来介绍KVC.         1.KVC(Key-Value Coding)键值编码             为了测试我们建立两个测试类                   测

iOS编程——Objective-C KVO/KVC机制[转]

这两天在看和这个相关的的内容,全部推翻重写一个版本,这是公司内做技术分享的文档总结,对结构.条理做了更清晰的调整.先找了段代码,理解下,网上看到最多的一段的关于KVC的代码 先上代码  1.     1 .Person类  2.     @implementation Person  3.     @synthesize name,age;//属性name 将被监视  4.     -(void) changeName  5.     {  6.         name=@"changeNam

老调重弹:对kvo的封装思路

关于kvo,kvo能做什么? kvo作为cocoa框架的重要特性之一,在底层框架中被大量使用.在特定的场合使用该特性往往能够带来难以想象的好处,让整个方案变得相当简洁和优雅.比如大名鼎鼎的下拉刷新的svpulltorefresh框架,其实现采用了category动态添加属性和kvo结合的方案,在egoRefresh框架的基础上获得了极大的改善,使调用者所要书写的代码量直接下降了一个量级.其中的奥秘在于通过kvo很好的处理了frame变化的问题,调用者不用再处理frame相关的代码,仅需要聚焦下拉

Objective-C中的老板是这样发通知的(Notification)

    ​    ​通知(Notification)简单的类比一下,公司的老总给下面的员工发通知啦,说明天公司要上市,各部门做一下准备工作.等通知发完,各部门收到后各司其职,做着自己该做的东西.假如Boss是通过公司的内部论坛发送的通知,那么Boss就是发送通知的对象,而公司员工就是通知的接受方,而公司的内部论坛就是通知中心,员要想接收到Boss发送的通知的前提是先在论坛上注册一下.老板只有一个,而接受通知的有多个部门,老板是通过论坛广播的形式发送的通知.说的高大上一点,通知是一种发送给一个或多

自定义Notification并利用Handler更新Notification

  在消息通知的时候,我们经常用到两个控件Notification和Toast.特别是重要的和需要长时间显示的信息,用Notification最合适不过了.他可以在顶部显示一个图标以标示有了新的通知,当我们拉下通知栏的时候,可以看到详细的通知内容. 最典型的应用就是未看短信和未接来电的显示,还有QQ企鹅,我们一看就知道有一个未接来电或者未看短信,正在挂QQ.同样,我们也可以自定义一个Notification来定义我们自己的程序想要传达的信息. Notification我把他分为两种,一种是默认的

手机卫士-安卓开发中的notification问题

问题描述 安卓开发中的notification问题 notification不能用, notification.setLatestEventInfo(context, "手机卫士", "保护中", pendingIntent); 代码中的setlatestEventinfo出错 很急,求大神解决 解决方案 http://www.2cto.com/kf/201502/374946.html 解决方案二: 应该是上下文没写好 context改为 this 或你定义的Ac

android notification 的总结分析_Android

分类  notification有以下几种: 1>普通notification 1.内容标题 2.大图标 3.内容 4.内容附加信息 5.小图标 6.时间 2>大布局Notification 图1 大布局notification是在android4.1以后才增加的,大布局notification与小布局notification只在'7'部分有区别,其它部分都一致.大布局notification只有在所有notification的最上 面时才会显示大布局,其它情况下显示小布局.你也可以用手指将其

全面解析Notification

Notification在Android中使用的频率可以说是非常高的,本文我将围绕着Notification的各方面进行解析,使大家对Notification有更好的认识. Notification的使用步骤 1.获取NotificationManager NotificationManager mNotificationManager = (NotificationManager) getSystemService(NOTIFICATION_SERVICE);  2.创建Notificatio