NSNotificationCenter使用说明

来源:互联网 发布:数据中心网络设计论文 编辑:程序博客网 时间:2024/05/22 08:24

  1. 定义一个方法

      -(void) update{       } 

2. 对象注册,并关连消息

     [[NSNotificationCenter defaultCenter] 

     addObserver:self selector:@selector(update) name:@"update" object:nil]  

3. 在要发出通知消息的地方

     [[NSNotificationCenter defaultCenter] 

     postNotificationName:@"update" object:nil];

具体如何使用 Notifications

用户可能使用RaiseMan并打开了几个document, 然后他发现紫色的背景颜色实在是不利于阅读文档正文. 

于是,他打开Preferences panel修改背景颜色,不过令人失望的是,已经存在的文档的背景颜色不会跟着改变. 

于是,这个用户可能会写信给你告诉你这些. 你也许会回复:"defualts会在document创建的时候才读取,

保存document在打开"实际上,用户想说明的是他希望程序能立马刷新已经打开的文档. 如果这样,那该怎么做呢?

我们需要把所有打开的document用一个list记录起来么?

--- 什么是Notification? ---

这个要求其实也很容易实现. 每个运行中的application都有一个NSNotificationCenter的成员变量,

它的功能就类似公共栏. 对象注册关注某个确定的notification(如果有人捡到一只小狗,就去告诉我). 

我们把这些注册对象叫做 observer. 其它的一些对象会给center发送notifications(我捡到了一只小狗). 

center将该notifications转发给所有注册对该notification感兴趣的对象. 我们把这些发送notification的对象叫做 poster

很多的标准Cocoa类会发送notifications: 在改变size的时候,Window会发送notification; 

选择table view中的一行时,table view会发送notification;我们可以在在线帮助文档中查看到标准cocoa对象发送的notification

在我们的例子中,我们将MyDocumet对象注册为observer. 而preference controller在用户改变color时将发送notification. 

MyDocument在接受到该notification后改变background color

在MyDocument对象释放前,我们必须从notification center移除我们注册的observer. 一般我们在dealloc方法中做这件事

-- Notifications 不是什么 --

当程序员们听到notification center的时候, 他们可能会联想到IPC(进程间通讯).他们认为:

"我在一个程序中创建一个observer,然后在另外一个程序中发送一个notification". 这个设计没有办法工作的, 

notification center允许同一个程序中的不同对象通许,它不能跨越不同的程序 [Notification 就是设计模 式中的 观察者模式, 

cocoa为我们实现了该模式, 就像Java也有同样的实现一样]

 

-- NSNotification 和 NSNotificationCenter

Notification对象非常简单. 它就是poster要提供给observer的信息包裹. notification对象有两个重要的成员变量: 

name 和 object. 一般object都是指向poster(为了让observer在接受到notification时可以回调到poster)

所以,notification有两个方法

    - (NSString *)name

    - (id)object

NSNotificaitonCernter是架构的大脑了.它允许我们注册observer对象, 发送notification, 撤销observer对象注册

下面是它的一些常用方法

+ (NSNotificationCenter *)defaultCenter

返回notification center [类方法,返回全局对象, 单件模式.cocoa的很多的全局对象都是通过类似方法实现]

- (void)addObserver:(id)anObserver

           selector:(SEL)aSelector

               name:(NSString *)notificationName

             object:(id)anObject

注册anObserver对象:接受名字为notificationName, 发送者为anObject的notification. 当anObject发送名字

为notificationName的notification时, 将会调用anObserver的aSelector方法,参数为该notification 如图14.1

. 如果notificationName为nil. 那么notification center将anObject发送的所有notification转发给observer

. 如果anObject为nil.那么notification center将所有名字为notificationName的notification转发给observer

 

- (void)postNotification:(NSNotification *)notification

发送notification至notification center 如图14.2

 

- (void)postNotificationName:(NSString *)aName

                      object:(id)anObject

创建并发送一个notification

 

- (void)removeObserver:(id)observer

移除observer

-- 发送一个Notification --

发送notification是其中最简单的步骤了,所以我们从它开始实现.当我们接收到changeBackgroundColor:消息时, 

PreferenceController对象发送一个notification.

 

我们将notification命名为@"BNRColorChanged" ,我们使用一个全局常量来指定.(有经验的程序员会使用一个前缀,

这样避免和其他组件定义的notification混淆)打开PreferenceController.h 添加下面的的外部申明

extern NSString * const BNRColorChangedNotification;

在PreferenceController.m中定义常量

NSString * const BNRColorChangedNotification = @"BNRColorChanged";

在PreferenceController.m修改changeBackgroundColor:方法

- (IBAction)changeBackgroundColor:(id)sender

{

    NSColor *color = [colorWell color];

    NSData *colorAsData =

                  [NSKeyedArchiver archivedDataWithRootObject:color];

    [[NSUserDefaults standardUserDefaults] setObject:colorAsData

                                          forKey:BNRTableBgColorKey];

 

    NSNotificationCenter *nc = [NSNotificationCenter defaultCenter];

    NSLog(@"Sending notification");

    [nc postNotificationName:BNRColorChangedNotification object:self];

}

-- 注册成为Observer --

要注册一个observer, 我们必须提供几个要数: 要成为observer的对象;所感兴趣的notification的名字;

当notification发送时要调用的方法. 我们也可以指定要关注莫个对象的notification.(比如说,我们需要

关注莫个特定的window的resize的notification)

 

编辑MyDocument类的init方法

- (id)init

{

    if (![super init])

        return nil;

 

    employees = [[NSMutableArray alloc] init];

 

    NSNotificationCenter *nc = [NSNotificationCenter defaultCenter];

    [nc addObserver:self

           selector:@selector(handleColorChange:)

               name:BNRColorChangedNotification

             object:nil];

    NSLog(@"Registered with notification center");

    return self;

}

同时在dealloc方法,将MyDocument从notification center中移除

- (void)dealloc

{

    [self setEmployees:nil];

    NSNotificationCenter *nc = [NSNotificationCenter defaultCenter];

    [nc removeObserver:self];

    [super dealloc];

}

-- 处理Notification --

当一个notification发生时, handleColorChange:方法将被调用. 目前我们在方法中简单的打印一些log.

- (void)handleColorChange:(NSNotification *)note

{

    NSLog(@"Received notification: %@", note);

}

编译运行程序,看到了我们想要的log了吧

-- userInfo Dictionary --

notification对象的object变量是poster,如果我们想要notification对象传递更多的信息,

我们可以使用user info dictionary. 每个notification对象有一个变量叫 userInfo, 它是一个NSDictionary对象,

用来存放用户希望随着notification一起传递到observer的其它信息. MyDocument将使用它来得到要改变的color.

在PreferenceController.m添加userInfo

- (IBAction)changeBackgroundColor:(id)sender

{

    NSColor *color = [sender color];

    NSData *colorAsData;

    colorAsData = [NSKeyedArchiver archivedDataWithRootObject:color];

    [[NSUserDefaults standardUserDefaults] setObject:colorAsData

                                          forKey:BNRTableBgColorKey];

 

    NSNotificationCenter *nc = [NSNotificationCenter defaultCenter];

    NSLog(@"Sending notification");

    NSDictionary *d = [NSDictionary dictionaryWithObject:color

                                                  forKey:@"color"];

    [nc postNotificationName:BNRColorChangedNotification

                      object:self

                    userInfo:d];

}

在MyDocument.m,从userInfo中读取到color

- (void)handleColorChange:(NSNotification *)note

{

    NSLog(@"Received notification: %@", note);

    NSColor *color = [[note userInfo] objectForKey:@"color"];

    [tableView setBackgroundColor:color];

}

打开几个窗口,并改变背景颜色,现在,那些打开的窗口的背景颜色立马就变了.

-- 思考 --

通常当你将自己的一个对象设置为cocoa某个标准对象的delegate的时候,你同时或许也对该标准对象的notification感兴趣. 

例如,我们实现一个window的delegate来处理 windowShouldClose: , 我们也许会对 NSWindowDidResizeNotification

这样的notification感兴趣.

如果一个cocoa标准对象有一个delegate,同时它也发送notification的话, cocoa对象会自动将它的delegate对象注册

成为observer来接受接受自己的notification. 如果我们实现了一个delegate,那么delegate[也就是我们的对象]

要怎样声明来接受notification呢?[方法的名字是什么?]

 

方法名字其实很简单: 以notification名字为基准, 先将NS前缀去掉,接着将第一个字母改为小写. 在将后面的Notification去掉,

然后加个冒号:. 例如,为了能接受到window的NSWindowDidResizeNotification, delegate可以实现方法:

- (void)windowDidResize:(NSNotification *)aNotification

 

当window改变大小时,这个方法将自动调用. 对于NSWindow,我们可以在.h或是帮助文档中找到类似的notification 

来实现notification方法.

-- 挑战 --

当程序不再是active状态是,让程序发出beep. 当unactive时,NSApplication会发送

NSApplicationDidResignActiveNotification的notificaiton. 而我们的AppController是NSApplication的delegate.   

函数NSBeep()可以用来发出beep声音
原创粉丝点击
热门问题 老师的惩罚 人脸识别 我在镇武司摸鱼那些年 重生之率土为王 我在大康的咸鱼生活 盘龙之生命进化 天生仙种 凡人之先天五行 春回大明朝 姑娘不必设防,我是瞎子 吃多了胃胀气怎么办 红薯吃多了胀气怎么办 裙子贴在腿上怎么办 薄裤子静电吸腿怎么办 雪纺衬衫起静电怎么办 吃烤饼不松软是怎么办 1岁宝宝睡眠不好怎么办 3岁幼儿睡眠不好怎么办 2岁幼儿睡眠不好怎么办 2岁宝宝睡眠不好怎么办 9岁儿童睡眠不好怎么办 3岁宝宝老踢被子怎么办 4岁宝宝老踢被子怎么办 四线锁边机跳线怎么办 引流管伤口洞红怎么办 甘蔗卡在喉咙里怎么办 棉花被子生虫了怎么办 绗缝羽绒服钻毛怎么办 宝珠笔没墨水了怎么办 衣服上画的笔印怎么办 黑笔芯弄衣服上怎么办 圆珠笔油在皮上怎么办 不小心吞了水银怎么办 小孩吃了洗发露怎么办? 脸上被铅笔戳了怎么办 小孩吃了铅笔芯怎么办 小孩把橡皮吃了怎么办 用棉签掏耳朵里面疼怎么办 棉签头掉耳朵里怎么办 黑裤子老是粘毛怎么办 新买的裤子掉色怎么办 黑裤子容易粘毛怎么办 裤子粘了全部毛怎么办 纯棉裤子粘毛了怎么办 裤子粘毛怎么办怎么洗 黑裤子洗白了怎么办 新买床单有味道怎么办 新买的床单扎人怎么办 刚买的衣服皱了怎么办 橘子沾到衣服上怎么办 橘子水掉衣服上怎么办