NSNotificationCenter

View Code
  1  1. 定义一个方法
2
3 -(void) update{ }
4
5 2. 对象注册,并关连消息
6
7 [[NSNotificationCenter defaultCenter]
8
9 addObserver:self selector:@selector(update) name:@"update" object:nil]
10
11 3. 在要发出通知消息的地方
12
13 [[NSNotificationCenter defaultCenter]
14
15 postNotificationName:@"update" object:nil];
16
17 具体如何使用 Notifications
18
19 http://blog.sina.com.cn/s/blog_5df7dcaf0100c0q2.html
20
21 ////////////////////////////////////////
22
23 第十四章: 使用 Notifications
24
25 用户可能使用RaiseMan并打开了几个document, 然后他发现紫色的背景颜色实在是不利于阅读文档正文.
26
27 于是,他打开Preferences panel修改背景颜色,不过令人失望的是,已经存在的文档的背景颜色不会跟着改变.
28
29 于是,这个用户可能会写信给你告诉你这些. 你也许会回复:"defualts会在document创建的时候才读取,
30
31 保存document在打开"实际上,用户想说明的是他希望程序能立马刷新已经打开的文档. 如果这样,那该怎么做呢?
32
33 我们需要把所有打开的document用一个list记录起来么?
34
35 --- 什么是Notification? ---
36
37 这个要求其实也很容易实现. 每个运行中的application都有一个NSNotificationCenter的成员变量,
38
39 它的功能就类似公共栏. 对象注册关注某个确定的notification(如果有人捡到一只小狗,就去告诉我).
40
41 我们把这些注册对象叫做 observer. 其它的一些对象会给center发送notifications(我捡到了一只小狗).
42
43 center将该notifications转发给所有注册对该notification感兴趣的对象. 我们把这些发送notification的对象叫做 poster
44
45 很多的标准Cocoa类会发送notifications: 在改变size的时候,Window会发送notification;
46
47 选择table view中的一行时,table view会发送notification;我们可以在在线帮助文档中查看到标准cocoa对象发送的notification
48
49 在我们的例子中,我们将MyDocumet对象注册为observer. 而preference controller在用户改变color时将发送notification.
50
51 MyDocument在接受到该notification后改变background color
52
53 在MyDocument对象释放前,我们必须从notification center移除我们注册的observer. 一般我们在dealloc方法中做这件事
54
55 -- Notifications 不是什么 --
56
57 当程序员们听到notification center的时候, 他们可能会联想到IPC(进程间通讯).他们认为:
58
59 "我在一个程序中创建一个observer,然后在另外一个程序中发送一个notification". 这个设计没有办法工作的,
60
61 notification center允许同一个程序中的不同对象通许,它不能跨越不同的程序 [Notification 就是设计模 式中的 观察者模式,
62
63 cocoa为我们实现了该模式, 就像Java也有同样的实现一样]
64
65
66
67 -- NSNotification 和 NSNotificationCenter
68
69 Notification对象非常简单. 它就是poster要提供给observer的信息包裹. notification对象有两个重要的成员变量:
70
71 name 和 object. 一般object都是指向poster(为了让observer在接受到notification时可以回调到poster)
72
73 所以,notification有两个方法
74
75 - (NSString *)name
76
77 - (id)object
78
79 NSNotificaitonCernter是架构的大脑了.它允许我们注册observer对象, 发送notification, 撤销observer对象注册
80
81 下面是它的一些常用方法
82
83 + (NSNotificationCenter *)defaultCenter
84
85 返回notification center [类方法,返回全局对象, 单件模式.cocoa的很多的全局对象都是通过类似方法实现]
86
87 - (void)addObserver:(id)anObserver
88
89 selector:(SEL)aSelector
90
91 name:(NSString *)notificationName
92
93 object:(id)anObject
94
95 注册anObserver对象:接受名字为notificationName, 发送者为anObject的notification. 当anObject发送名字
96
97 为notificationName的notification时, 将会调用anObserver的aSelector方法,参数为该notification 如图14.1
98
99 . 如果notificationName为nil. 那么notification center将anObject发送的所有notification转发给observer
100
101 . 如果anObject为nil.那么notification center将所有名字为notificationName的notification转发给observer
102
103
104
105 - (void)postNotification:(NSNotification *)notification
106
107 发送notification至notification center 如图14.2
108
109
110
111 - (void)postNotificationName:(NSString *)aName
112
113 object:(id)anObject
114
115 创建并发送一个notification
116
117
118
119 - (void)removeObserver:(id)observer
120
121 移除observer
122
123 -- 发送一个Notification --
124
125 发送notification是其中最简单的步骤了,所以我们从它开始实现.当我们接收到changeBackgroundColor:消息时,
126
127 PreferenceController对象发送一个notification.
128
129
130
131 我们将notification命名为@"BNRColorChanged" ,我们使用一个全局常量来指定.(有经验的程序员会使用一个前缀,
132
133 这样避免和其他组件定义的notification混淆)打开PreferenceController.h 添加下面的的外部申明
134
135 extern NSString * const BNRColorChangedNotification;
136
137 在PreferenceController.m中定义常量
138
139 NSString * const BNRColorChangedNotification = @"BNRColorChanged";
140
141 在PreferenceController.m修改changeBackgroundColor:方法
142
143 - (IBAction)changeBackgroundColor:(id)sender
144
145 {
146
147 NSColor *color = [colorWell color];
148
149 NSData *colorAsData =
150
151 [NSKeyedArchiver archivedDataWithRootObject:color];
152
153 [[NSUserDefaults standardUserDefaults] setObject:colorAsData
154
155 forKey:BNRTableBgColorKey];
156
157
158
159 NSNotificationCenter *nc = [NSNotificationCenter defaultCenter];
160
161 NSLog(@"Sending notification");
162
163 [nc postNotificationName:BNRColorChangedNotification object:self];
164
165 }
166
167 -- 注册成为Observer --
168
169 要注册一个observer, 我们必须提供几个要数: 要成为observer的对象;所感兴趣的notification的名字;
170
171 当notification发送时要调用的方法. 我们也可以指定要关注莫个对象的notification.(比如说,我们需要
172
173 关注莫个特定的window的resize的notification)
174
175
176
177 编辑MyDocument类的init方法
178
179 - (id)init
180
181 {
182
183 if (![super init])
184
185 return nil;
186
187
188
189 employees = [[NSMutableArray alloc] init];
190
191
192
193 NSNotificationCenter *nc = [NSNotificationCenter defaultCenter];
194
195 [nc addObserver:self
196
197 selector:@selector(handleColorChange:)
198
199 name:BNRColorChangedNotification
200
201 object:nil];
202
203 NSLog(@"Registered with notification center");
204
205 return self;
206
207 }
208
209 同时在dealloc方法,将MyDocument从notification center中移除
210
211 - (void)dealloc
212
213 {
214
215 [self setEmployees:nil];
216
217 NSNotificationCenter *nc = [NSNotificationCenter defaultCenter];
218
219 [nc removeObserver:self];
220
221 [super dealloc];
222
223 }
224
225 -- 处理Notification --
226
227 当一个notification发生时, handleColorChange:方法将被调用. 目前我们在方法中简单的打印一些log.
228
229 - (void)handleColorChange:(NSNotification *)note
230
231 {
232
233 NSLog(@"Received notification: %@", note);
234
235 }
236
237 编译运行程序,看到了我们想要的log了吧
238
239 -- userInfo Dictionary --
240
241 notification对象的object变量是poster,如果我们想要notification对象传递更多的信息,
242
243 我们可以使用user info dictionary. 每个notification对象有一个变量叫 userInfo, 它是一个NSDictionary对象,
244
245 用来存放用户希望随着notification一起传递到observer的其它信息. MyDocument将使用它来得到要改变的color.
246
247 在PreferenceController.m添加userInfo
248
249 - (IBAction)changeBackgroundColor:(id)sender
250
251 {
252
253 NSColor *color = [sender color];
254
255 NSData *colorAsData;
256
257 colorAsData = [NSKeyedArchiver archivedDataWithRootObject:color];
258
259 [[NSUserDefaults standardUserDefaults] setObject:colorAsData
260
261 forKey:BNRTableBgColorKey];
262
263
264
265 NSNotificationCenter *nc = [NSNotificationCenter defaultCenter];
266
267 NSLog(@"Sending notification");
268
269 NSDictionary *d = [NSDictionary dictionaryWithObject:color
270
271 forKey:@"color"];
272
273 [nc postNotificationName:BNRColorChangedNotification
274
275 object:self
276
277 userInfo:d];
278
279 }
280
281 在MyDocument.m,从userInfo中读取到color
282
283 - (void)handleColorChange:(NSNotification *)note
284
285 {
286
287 NSLog(@"Received notification: %@", note);
288
289 NSColor *color = [[note userInfo] objectForKey:@"color"];
290
291 [tableView setBackgroundColor:color];
292
293 }
294
295 打开几个窗口,并改变背景颜色,现在,那些打开的窗口的背景颜色立马就变了.
296
297 -- 思考 --
298
299 通常当你将自己的一个对象设置为cocoa某个标准对象的delegate的时候,你同时或许也对该标准对象的notification感兴趣.
300
301 例如,我们实现一个window的delegate来处理 windowShouldClose: , 我们也许会对 NSWindowDidResizeNotification
302
303 这样的notification感兴趣.
304
305 如果一个cocoa标准对象有一个delegate,同时它也发送notification的话, cocoa对象会自动将它的delegate对象注册
306
307 成为observer来接受接受自己的notification. 如果我们实现了一个delegate,那么delegate[也就是我们的对象]
308
309 要怎样声明来接受notification呢?[方法的名字是什么?]
310
311
312
313 方法名字其实很简单: 以notification名字为基准, 先将NS前缀去掉,接着将第一个字母改为小写. 在将后面的Notification去掉,
314
315 然后加个冒号:. 例如,为了能接受到window的NSWindowDidResizeNotification, delegate可以实现方法:
316
317 - (void)windowDidResize:(NSNotification *)aNotification
318
319
320
321 当window改变大小时,这个方法将自动调用. 对于NSWindow,我们可以在.h或是帮助文档中找到类似的notification
322
323 来实现notification方法.
324
325 -- 挑战 --
326
327 当程序不再是active状态是,让程序发出beep. 当unactive时,NSApplication会发送
328
329 NSApplicationDidResignActiveNotification的notificaiton. 而我们的AppController是NSApplication的delegate.
330
331 函数NSBeep()可以用来发出beep声音

posted on 2012-02-21 08:57  AXZON  阅读(317)  评论(0编辑  收藏  举报

导航