2019-03-13 17:21:49 qq_41856760 阅读数 380

一、通知

通知是iOS中的一种消息传递方式,通过消息中心(NSNotificationCenter)对消息的监听,当某些类发送出消息的时候,消息中心监听到这些消息,然后进行相应的操作,这些操作对于发送出这些消息的类来说是相同的。

下面通过一个demo来说明通知的实现

                                 

就是点击按钮,弹出弹窗,然后点击弹窗中的按钮打印出一些信息。 这个信息是由前面自定义的alertView发出给到控制器的,我们用通知来实现这之间的数据传递。


//viewController.m
- (IBAction)buttonDidClicked:(UIButton *)sender {
    
    LSRAlertView * view = [LSRAlertView alertWithTitle:@"操作完成" andTitleImageName:@"gou"];
    [view addTopButtonWithTitle:@"确定"];
    [view addBottomButtomWithTitle:@"取消"];
    [view show];
    //这里向消息中心注册消息
    //obsever:观察者,谁来监听这个消息
    //selector:监听到发出的对应的消息后要做什么
    //name:监听的消息的名称
    //obeject:保存在消息中心的数据,一般传一个nil
    [[NSNotificationCenter defaultCenter]addObserver:self selector:@selector(sureButtonDidClicked) name:@"SureButtonClickedNotificationName" object:nil];
    //取消按钮我们需要传递一些参数
    [[NSNotificationCenter defaultCenter]addObserver:self selector:@selector(cancelButtonClicked:) name:@"CancelButtonClickedNotificationName" object:nil];
    
}

首先在控制器中添加一个按钮点击事件,然后设置弹窗视图(至于如何自定义弹窗视图,请查看我的上一篇文章),接着我们在这里向消息中心注册消息,为什么在这里注册呢?因为我们是要在控制器中获取到弹窗视图的数据,在控制器中注册相应的消息后,控制器就会监听发出我们注册消息的对象,然后执行相应的方法,所以在控制器中注册消息。

接着我们在自定义弹窗视图的类写发送消息的代码,那么又该在哪里写呢?应该在弹窗视图上的按钮被点击了之后写,所以:


//LSRAlertView.m
- (IBAction)topButonClicked:(UIButton *)sender {
     [self dismiss];
    //这里按钮被点击了,此时需要发出消息
    //name:消息的名称,必须和所监听的消息的名称相同
    //object:需要回调的数据
    //userInfo:也是需要回调的数据
    [[NSNotificationCenter defaultCenter]postNotificationName:@"SureButtonClickedNotificationName" object:nil userInfo:nil];
   
}
- (IBAction)bottomButtonClicked:(UIButton *)sender {
    //发出消息的时候回调参数
    [self dismiss];
    [[NSNotificationCenter defaultCenter]postNotificationName:@"CancelButtonClickedNotificationName" object:@"object" userInfo:@{@"name":@"jack"}];
    
}

发送消息我们用postNotificationName,这里需要注意的是发送消息的名称一定要和之前在控制器中注册的消息名对应起来,另外如果我们需要传递一些数据,就将传递的数据写在 object 或者  userInfo 中,前者是一个字符串类型,后者是一个字典类型。

然后是我们在控制器监听到弹窗视图发出的消息后执行的操作:

//ViewController.m
-(void)sureButtonDidClicked{
    //这里不需要回调参数
    NSLog(@"确定按钮被点击了");
    
    //在这里移除消息
    [[NSNotificationCenter defaultCenter]removeObserver:self name:@"SureButtonClickedNotificationName" object:nil];
} 
-(void)cancelButtonClicked:(NSNotification *)notification{
    //这里将传递过来的参数打印出来
    NSLog(@"%@",notification.object);
    NSLog(@"%@",notification.userInfo);
    
    //同样在这里移除消息
    [[NSNotificationCenter defaultCenter]removeObserver:self name:@"CancelButtonClickedNotificationName" object:nil];
    
}

这里需要注意两点,一是如何取出消息中的数据,我们使用notification的属性object和userInfo来取得相应的数据。第二点也是非常重要的就是一定要将消息中的消息移除,一定要将消息中的消息移除,一定要将消息中的消息移除。如果不移除就会导致点击一次按钮触发多次事件,因为有多个相同的消息同时注册了。

最后是打印的信息:

2017-12-13 16:27:51 w582324909 阅读数 1361

1、通过一个简单的小程序(不考虑程序的实用逻辑,纯粹为了掩饰)来介绍通知的使用,先上图:

点击注册进入注册界面,然后填写用户名,点击save的时候,用户名传值到首页并显示


2、原理:利用通知,这里的通知是广播通知,可以说是一对多,意思就是一个广播向所有人发出一个消息,而接受者不在乎是谁发的,能接收到就行。当输入用户名点击save的时候,这时候就发送一个通知,让需要接收通知的人知道,这里首页需要知道用户名是什么,所以首页的controller.m中应该有一段接收通知的代码,这样一个发一个接,就能把信息传递出去了。(通知还有其他几种类型,这里就不过多介绍。)


3、上代码:

(1)投送通知:

这里应该在注册的viewcontroller里写上投送通知,谁要传值,就在哪里写投送

NSDictionary *dataDic = [NSDictionary dictionaryWithObject:self.username.text forKey:@"username"];
[[NSNotificationCenter defaultCenter] postNotificationName:@"RegisterCompletionNotification" object:nil userInfo:dataDic];

划重点postNotificationName,大家看到post就应该知道这个是投送通知的代码了。


(2)注册接收通知:

这里主页需要接收通知,所以这段代码应该写在主页的viewcontroller里

- (void)viewDidLoad {
    [super viewDidLoad];
    [[NSNotificationCenter defaultCenter] addObserver:self selector:@selector(registerCompletion:) name:@"RegisterCompletionNotification" object:nil];  //这里就是监听通知的代码,会调用下面注册通知,注册通知后,有符合监听规则的通知发过来,通知就会被监听到
}

-(void)registerCompletion:(NSNotification*)notification {
    NSDictionary *theData = [notification userInfo];
    NSString *thename = [theData objectForKey:@"username"];
    self.username.text = thename;
    NSLog(@"%@",thename);
}

划重点registerCompletion,看到register,应该可以理解这个是注册通知的代码了


- (void)didReceiveMemoryWarning {
    [super didReceiveMemoryWarning];
    [[NSNotificationCenter defaultCenter]removeObserver:self];  //解除注册通知
}

划重点:

@”RegisterCompletionNotification“  和@username前后(两个控制器间)一定要写对了

前后要一致,具体怎么命名看个人习惯,最好规范易懂一点


4、最后说一句:

博主时隔很久才上CSDN更新博客,重拾之前放下的OC,如果有什么写的有误的地方或者需要改进的地方,请在下面留言,谢谢。

现在就当做一个新人,重新学习OC吧。





2015-12-01 15:47:43 haoxindasoft 阅读数 248

注:一个应用程序中只有一个通知中心


/// @brief 创建一个通知中心对象

NSNotificationCenter  *nc = [NSNotificationCenter  defaultCenter];


/// @brief 向通知中心注册观察者,第一个参数为观察的对象所在类,第二个参数为观察到以后所调用的方法,第三个参数为观察的事件,第四个参数为要传的参数

[nc addObserver:self  selector:@selector(keyboardShowUp) name:UIKeyboardWillShowNotification  object:nil];


/// @brief 发送通知,第三个参数为发送的消息

[[NSNotificationCenter defaultCenter] postNotifiationName:@"发送的通知名"  object:self  userInfo:dict];


/// @brief 获取通知消息,noti为通知中心的对象名

NSDictionary  *dict = noti.userInfo


/// @brief 从通知中心移除观察者

[[NSNotificationCenter  defaultCenter]  removeObserver:self];

2013-12-09 13:57:24 u012894631 阅读数 772

1. 定义一个方法

      -(void) update{       } 

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

     [[NSNotificationCenter defaultCenter] 

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

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

     [[NSNotificationCenter defaultCenter] 

     postNotificationName:@"update" object:nil];

具体如何使用 Notifications

http://blog.sina.com.cn/s/blog_5df7dcaf0100c0q2.html

////////////////////////////////////////

第十四章: 使用 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声音

2016-06-04 13:47:26 qq_27325349 阅读数 24925

在实际的开发中,通知这一手段还是比较常用的。那今天就详细讲讲通知。

1.通知中心(NSNotificationCenter)实际是在程序内部提供了一种广播机制。把接收到的消息,根据内部的消息转发表,将消息转发给需要的对象。这句话其实已经很明显的告诉我们要如何使用通知了。第一步:在需要的地方注册要观察的通知,第二步:在某地方发送通知。(这里注意:发送的通知可能是我们自定义的,也可能是系统的)。


2. 具体如何使用通知:

比如在某个控制器中需要观察某通知,那么代码如下:




在需要发送通知的地方代码如下:



已经是一目了然了。但是还没结束,千万不要忘了在注册观察通知的控制器的dealloc方法中移除要观察的通知。这是因为,当控制器因为某些原因比如内存问题而被销毁的时候,通知中心被注册的该通知还是存在的,而当其他有地方发送该通知的时候,通知中心会继续转发,但是转发的对象已经不存在了,这时候就会crash了。


所以说,良好的编码习惯还是很重要的,能够避免一些不必要的麻烦。


3.有些同学可能习惯会在viewWillAppear和viewWillDisappear方法中配对使用,比如这样:


也无可厚非,不过这两个方法属于会被“经常调用”的方法,比如左滑右滑的时候会重复调用,还是要多考虑一下业务逻辑,避免不必要的坑。


个人还是建议在viewDidLoad中注册通知,在dealloc中移除通知。

==》有些同学可能会发现,在某个ViewController中,没有移除通知好像也没有发生过crash。是的,这是因为在viewController被销毁的时候,它帮我们调用了remove方法。想去证明的同学可以新建一个分类,覆盖系统的removeObserver方法,会发现在controller销毁的时候调用了我们重写的remove方法。

iOS通知的介绍

阅读数 252

IOS学习笔记之通知

阅读数 298

iOS中的通知机制

阅读数 236

iOS通知的使用

阅读数 272

iOS通知

博文 来自: u012121216
没有更多推荐了,返回首页