通知对象可以携带数据,然后通过通知中心(notification center)进行广播,若想要响应这个通知,接受者需要注册监听这个通知。通知机制的工作原理给我们一个机会,我们可以利用通知机制,达到解藕程序的目的。我们可以把模块的通用功能封装成组件,与业务相关的功能,可以交给组件的使用者实现。组件和组件使用者之间的通信恰好可以使用通知机制来完成。iOS中有三种类型的通知,普通通知,本地通知,以及远程通知。本文讲解的是normal notification,也就普通通知,他是一个NSNotification类型的对象。
发送通知的过程:首先,我们需要构造一个通知对象NSNotification,然后通过通知中心NSNotificationCenter的实例方法postNotification广播出去。
头文件
static NSString *const kNotificationName=@"kNotificationName"; @interface ViewController : UIViewController @end
发送通知实现代码
NSNotification *notification=[NSNotification notificationWithName:kNotificationName object:self userInfo:@{@"key1":@"value1", @"key2":@"value2"}]; [[NSNotificationCenter defaultCenter] postNotification:notification];
Name
这是一个字符串,我们必须为每一个通知指定一个名字,当我们监听通知的时候,需要指明通知的名字。因此,当我们自定义通知的时候,在定义通知名字的头件中详细描述通知的用处,发送对象是谁,包含哪些数据。我们可以参考一下系统文件UIWindow.h中键盘显隐相关的通知的定义。
// Each notification includes a nil object and a userInfo dictionary containing the // begining and ending keyboard frame in screen coordinates. Use the various UIView and // UIWindow convertRect facilities to get the frame in the desired coordinate system. // Animation key/value pairs are only available for the "will" family of notification. UIKIT_EXTERN NSString *const UIKeyboardWillShowNotification; UIKIT_EXTERN NSString *const UIKeyboardDidShowNotification; UIKIT_EXTERN NSString *const UIKeyboardWillHideNotification; UIKIT_EXTERN NSString *const UIKeyboardDidHideNotification;Sender object
Sender object 正如其字面意思,指明哪个对象发送通知。通常它的值是self。为什么我们需要指明到底是哪个对象发送通知?比如这样一个场景,应用程序在class1中发送了名字为MyNotification的通知,而且在另一个类class2中发送了一个相同名字的通知,如果接受者A只想接受class1发送的通知MyNotification,不想接受class2的通知,这时候我们在注册通知的时候,指明通知源头为class1就可以了。
User info dictionary
你可以存储一些key/value键值对到字典中。你能过通过user info 传递一些重要信息给接受者。
调用NSNotificationCenter的addObserver:selector:name:object方法监听通知。这个方法的参数
addObserver
observer确定谁来观察通知,如果是当前所在的类的话,传递self即可
selector
selector方法接受通知notification,此方法必须有类型为NSNotification的参数
name
通知的名字
object
发送通知的对象
监听通知代码实现
[[NSNotificationCenter defaultCenter] addObserver:self selector:@selector(responseNotification:) name:kNotificationName object:[[ViewController alloc] init]];
我们创建一个简单的单视图程序,在屏幕的底部有一个textfield文本输入框,当我们点击textfield输入文字的时候,软键盘会弹出来,而且完完全全地遮住了textfield控件。
#import "ViewController.h" @interface ViewController () @property (weak, nonatomic) IBOutlet UIScrollView *scrollView; @property (weak, nonatomic) IBOutlet UITextField *textField; @property (weak, nonatomic) IBOutlet UILabel *label; @end在view controller的viewWillAppear:方法中添加监听UIKeyboardWillShowNotification键盘将显示通知和UIKeyboardWillHideNotification键盘将隐藏通知
- (void)viewWillAppear:(BOOL)animated{ [super viewWillAppear:animated]; [[NSNotificationCenter defaultCenter] addObserver:self selector:@selector(handleKeyboardWillShow:) name:UIKeyboardWillShowNotification object:nil]; [[NSNotificationCenter defaultCenter] addObserver:self selector:@selector(handleKeyboardWillHide:) name:UIKeyboardWillHideNotification object:nil]; }
- (void)viewWillDisappear:(BOOL)animated{ [super viewWillDisappear:animated]; [[NSNotificationCenter defaultCenter] removeObserver:self]; }注意:这里有个我们常犯的错误,在viewDidLoad中开始监听通知,和在dealloc中移除监听。这会产生一些问题。我们的视图A从屏幕消失,而另一个视图B显示在屏幕中,点击B的文本输入框,系统会发出UIKeyboardWillShowNotification的通知,此时,视图A仍然能够监听到通知,因为它只是暂时地从屏幕移除,并没有被销毁,所以不会调用dealloc方法中的移除监听的操作。
- (void)handleKeyboardWillShow:(NSNotification *)notification{ //获取键盘显示过程的动画时间,以及键盘出现在屏幕中的坐标 NSValue *animationDurationObject=[notification.userInfo objectForKey:UIKeyboardAnimationDurationUserInfoKey]; NSValue *keyboardFrameEndObject=[notification.userInfo objectForKey:UIKeyboardFrameEndUserInfoKey]; double animationDuration=0; CGRect keyboardFrameEnd=CGRectZero; [animationDurationObject getValue:&animationDuration]; [keyboardFrameEndObject getValue:&keyboardFrameEnd]; //将frame从window坐标系统转换到我们的视图坐标系统 UIWindow *window=[[UIApplication sharedApplication] keyWindow]; CGRect convertedKeyboardFrameEnd=[self.view convertRect:keyboardFrameEnd fromView:window]; //获取键盘遮盖视图的frame CGRect intersectionOfKeyboardAndView=CGRectIntersection( self.view.frame,convertedKeyboardFrameEnd); //滚动scroll view [UIView animateWithDuration:animationDuration animations:^{ self.scrollView.contentInset=UIEdgeInsetsMake(0.0f, 0.0f, intersectionOfKeyboardAndView.size.height, 0.0f); }]; }
- (void)handleKeyboardWillHide:(NSNotification *)notification{ NSValue *animationDurationObject=[notification.userInfo objectForKey:UIKeyboardAnimationDurationUserInfoKey]; double animationDuration=0; [animationDurationObject getValue:&animationDuration]; [UIView animateWithDuration:animationDuration animations:^{ self.scrollView.contentInset=UIEdgeInsetsZero; }]; }
- (BOOL)textFieldShouldReturn:(UITextField *)textField{ //注销text field的作为第一响应对象,隐藏键盘 [self.textField resignFirstResponder]; return YES; }