深入剖析通知中心和KVO
深入剖析通知中心和KVO
要先了解KVO和通知中心,就得先说说观察者模式,那么观察者模式到底是什么呢?下面来详细介绍什么是观察者模式。
观察者模式
-A对B的变化感兴趣,就注册成为B的观察者,当B发生变化时通知A,告知B发生了变化,这就是观察者模式。
观察者模式定义了一对一对多的依赖关系,让多个观察者对象同时监听某一个主题对象,这个主题对象在状态发生变化时,会通知所有的观察者对象,使他们能够自动更新自己或者作出相应的一些动作。
在开发中,我们可能会接触到观察者模式的实现方式,有NSNotificationCenter、KVO等。
下面就来上正菜了,在下面我会给大家详细讲解通知中心和KVO到底是什么,对于这两个,我也是模模糊糊,搞不清到底是什么东西,所以才找了资料,来整理整理,顺便写篇博客分享给大家。
通知中心
- 通知中心就是以NSNotificationCenter为中心,观察者往Center中注册对某个主题对象的变化感兴趣,主体对象通过NSNotificationCenter进行变化广播,所有的变化和监听行为都向同一个中心进行注册,所有对象的变化也通过同一个中心对外广播。

通知中心的执行顺序
通过NSNotificationCenter的addObserver:selector:name:object接口来注册通知中心
被观察的对象,通过postNotificationName:object:userInfo:向通知中心发送某一类型通知。
当有通知来的时候,Center会调用观察者注册的接口来广播通知,同时传递存储着更改内容的NSNotification对象。
当该通知不再使用时,可以在dealloc方法里removeObserver:删除观察者
代码
main.m
//
// main.m
// 通知中心
//
// Created by ma c on 16/5/16.
// Copyright © 2016年 Jujue. All rights reserved.
//
#import <Foundation/Foundation.h>
#import "Observer.h"
#import "HttpClient.h"
int main(int argc, const char * argv[]) {
@autoreleasepool {
// insert code here...
//1、通知中心注册通知对象
//2、需要给通知中心发送消息的地方post通知
//3、执行注册通知时设置的方法
Observer *observer = [[Observer alloc]init];
//在通知中心注册成为Observer的观察者
[observer registNotification];
HttpClient *client = [[HttpClient alloc]init];
//发送消息
[client postNotifi];
}
return 0;
}
Observer.m
//
// Observer.m
// 通知中心
//
// Created by ma c on 16/5/16.
// Copyright © 2016年 Jujue. All rights reserved.
//
#import "Observer.h"
@implementation Observer
- (void)registNotification
{
NSNotificationCenter *notificationCenter = [NSNotificationCenter defaultCenter];
//注册成为观察者 观察对象是Observer
[notificationCenter addObserver:self selector:@selector(doAction:) name:@"Observer" object:nil];
}
- (void)doAction:(NSNotification *)notifi
{
//发送者是哪个类就输出的是什么东西-这个例子打印出来是 <HttpClient: 0x10020ead0>
NSLog(@"%@",notifi.object);
//打印结果 name = xiaoyu;
NSLog(@"%@",notifi.userInfo);
}
- (void)dealloc
{
[[NSNotificationCenter defaultCenter]removeObserver:self];
}
@end
HttpClient.m
//
// HttpClient.m
// 通知中心
//
// Created by ma c on 16/5/16.
// Copyright © 2016年 Jujue. All rights reserved.
//
#import "HttpClient.h"
@implementation HttpClient
- (void)postNotifi
{
NSDictionary *info = @{@"name":@"xiaoyu"};
NSNotificationCenter *notificationCenter = [NSNotificationCenter defaultCenter];
[notificationCenter postNotificationName:@"Observer" object:self userInfo:info];
}
@end
KVO(键值观察)
KVO是一个典型的观察者模式,观察者在键值改变时会得到通知。iOS中有个Notification的机制,也可以获得通知,但这个机制需要有个Center,相比之下KVO更加简洁而直接。
NSKeyValueObserving非正式协议定义了一种机制,它允许对象去监听其它对象的某个属性的修改。NSObject提供了一个NSKeyValueObserving协议的默认实现,它为所有对象提供了一种自动发送修改通知的能力。我们可以通过禁用自动发送通知并使用这个协议提供的方法来手动实现通知的发送,以便更精确地去处理通知。
KVO的实现步骤
(1)注册需要观察的对象的属性addObserver:forKeyPath:options:context:
(2)实现observeValueForKeyPath:ofObject:change:context:方法,这个方法当观察的属性变化时会自动调用.在这个方法中还通过NSKeyValueObservingOptionNew这个参数要求把新值在dictionary中传递过来。
(3)取消注册观察removeObserver:forKeyPath:context:
手动设置KVO
//关闭某key值的KVO + (BOOL)automaticallyNotifiesObserversForKey:(NSString *)key { BOOL automatic = YES;
if ([key isEqualToString:@"age"]) {
automatic = NO;
} else {
automatic = [super automaticallyNotifiesObserversForKey:key];
}
return automatic;
} //重写setter方法手动设置KVO - (void)setAge:(int)age { //手动设置KVO if (_age != age) { [self willChangeValueForKey:@"age"]; _age = age; [self didChangeValueForKey:@"age"]; }
}
注意:一般的建议是,在获取属性值时,可以用实例变量,在设置属性值时,尽量用setter方法,以保证属性的KVO特性。
KVO的实现原理
- 当某一个类的实例第一次使用KVO的时候,系统就会在运行期间动态的创建该类的一个派生类,该类的命名规则一般是以NSKVONotifying为前缀,以原本的类名为后缀。并且将原型的对象的isa指针指向该派生类。同时在派生类中重载了使用KVO的属性的setter方法,在重载的setter方法中实现真正的通知机制,正如前面我们手动实现KVO一样。这么做是基于设置属性会调用 setter 方法,而通过重写就获得了 KVO 需要的通知机制。当然前提是要通过遵循 KVO 的属性设置方式来变更属性值,如果仅是直接修改属性对应的成员变量,是无法实现 KVO 的。
代码-大家可以去我的github去下载
深入剖析通知中心和KVO的更多相关文章
- 通知中心NSNotificationCenter的使用
通知中心NSNotificationCenter的使用 Cocoa框架中,通知中心以及KVO都属于设计模式中的观察者. Source 在使用通知中心之前,对通知中心类进行了简单的封装,以便可读性更强. ...
- OC 观察者模式(通知中心,KVO)
OC 观察者模式(通知中心,KVO) 什么是观察者模式??? A对B的变化感兴趣,就注册为B的观察者,当B发生变化时通知A,告知B发生了变化.这就是观察者模式. 观察者模式定义了一种一对多的依赖关系, ...
- KVO和通知中心
苹果其实在语言层面为我们做了很多事,比如Category实现装饰模式的意图,target-action实现命令模式意图等等,对于观察者模式,苹果也提供了KVO和通知中心,给开发者提供了极大的便利. 观 ...
- iOS之NSNotificationCenter通知中心使用事项
其实这里的通知和之前说到的KVO功能很想,也是用于监听操作的,但是和KVO不同的是,KVO只用来监听属性值的变化,这个发送监听的操作是系统控制的,我们控制不了,我们只能控制监听操作,类似于Androi ...
- 通知中心 NSNotificationCenter 的简单使用方法
NSNotificationCenter(通知中心) [注意]需再dealloc中移除观察者 获取通知中心单例对象 NSNotificationCenter *center=[NSNotifi ...
- iOS通知中心升级 -可设置按优先级执行block
简单介绍下,这是需求驱动中发现iOS的NotificationCenter有很多功能无法实现,于是对其进行了一层包装.相当于手动管理观察者栈和监听者期望执行的事件,因此可以为其添加了很多新增的功能,将 ...
- iOS开发中的错误整理,再一次整理通过通知中心来处理键盘,一定记得最后关闭通知中心
一.打开通知中心,监听键盘的显示与隐藏 二.最后记得将监听通知的对象移除
- iOS中通知中心(NSNotificationCenter)的使用总结
一.了解几个相关的类 1.NSNotification 这个类可以理解为一个消息对象,其中有三个成员变量. 这个成员变量是这个消息对象的唯一标识,用于辨别消息对象. @property (readon ...
- 使用 Swift 制作一个新闻通知中心插件(2)
我们在第一部分的文章中详细讲解了创建一个通知中心插件的整体过程.我们成功的在通知中心里面显示了新闻列表.但是截止到目前,我们还不能从通知中心的列表中查看新闻的详细内容.在这次的教程中,我们就以上次的教 ...
随机推荐
- MVC解决Json DataGrid返回的日期格式是/Date(20130450000365)
实际上是Json格式化问题,我们应该在返回json的时候进行格式化,我们需要重写系统的JsonResult类 using System; using System.Collections.Generi ...
- 外网访问原理分析 - 每天5分钟玩转 OpenStack(105)
本节我们会将上节创建的 ext_net 连接到 router,并验证内外网的连通性. 更重要的,我们会分析隐藏在表象之下的原理. 将外网连接到 Neutron 的虚拟路由器,这样 instance 才 ...
- 细说Linq之Aggregate
前言 Linq中有关常见的方法我们已经玩的得心应手,而对于那些少用的却是置若罔闻(夸张了点),但只有在实际应用中绞尽脑汁想出的方法还不如内置的Linq方法来的实际和简洁,不喜勿喷,怪我见识短. 通过R ...
- 细说gulp
一.概述&安装 Gulp,简而言之,就是前端自动化开发工具,利用它,我们可以提高开发效率. 比如: 1. 压缩js 2. 压缩css 3. 压缩less 4. 压缩图片 等等… 我们完 ...
- 移动端用js与jquery实时监听输入框值的改动
背景: 在一次移动端H5开发中,需要监听输入框值的实时变动. onchange事件肯定抛弃,因为只能失去焦点才触发. 而keyPress在Android可以触发,iOS不可以. 又不想用Android ...
- 编写高质量代码:改善Java程序的151个建议(第1章:JAVA开发中通用的方法和准则___建议16~20)
建议16:易变业务使用脚本语言编写 Java世界一直在遭受着异种语言的入侵,比如PHP,Ruby,Groovy.Javascript等,这些入侵者都有一个共同特征:全是同一类语言-----脚本语言,它 ...
- Struts2 源码分析——拦截器的机制
本章简言 上一章讲到关于action代理类的工作.即是如何去找对应的action配置信息,并执行action类的实例.而这一章笔者将讲到在执行action需要用到的拦截器.为什么要讲拦截器呢?可以这样 ...
- 【原创】kafka consumer源代码分析
顾名思义,就是kafka的consumer api包. 一.ConsumerConfig.scala Kafka consumer的配置类,除了一些默认值常量及验证参数的方法之外,就是consumer ...
- C#播放wav文件
C#使用HWQPlayer类播放wav文件 类的代码: using System.IO; using System.Runtime.InteropServices; namespace HoverTr ...
- Entity Framework 6 开发系列 目录
2014 年开始接触 Entity Framework 6 也快两年,用它已经沉淀了一个成熟架构,也用来开发了不少大大小小的产品和项目,直到这段时间,才真正有时间来回顾,重新学习它,为让大家更加了解E ...