详细的KVO总结,包括基本改变,使用案例,注意点.看我就够了!
概述
- KVO全称Key-Value-Observing,也叫键值监听,是一种观察者设计模式.提供了一种机制,当指定的对象的属性被修改后,对象就会收到一个通知.也就是说每次指定的被观察的对象的属性被修改后,KVO就会自动通知相应的观察者.
- 优势:可以降低两个类(业务逻辑和视图控制的类)之间的耦合性.也就是说可以很容易的实现视图组件和数据模型的分离.当数据模型的属性值改变之后作为监听器的视图组件就会被激发,激发时就会回调监听器自身.
- 在Objective-C中要实现KVO则必须实现NSKeyValueObServing协议.但不用担心,因为NSObject已经实现了该协议,因此几乎所有的Objective-C对象都可以使用KVO.
KVO的方法
- 监听方法
- (void)addObserver:(NSObject *)observer forKeyPath:(NSString *)keyPath options:(NSKeyValueObservingOptions)options context:(nullable void *)context;
某一个对象(受虐狂,喜欢被人监视),给自己添加一个监听者(一般都是控制器本身self),让监听者监听自身的某一个属性. options就是要求监听者记录的信息. context就是要监听者给自己添加一个标记,以防止和别的对象的监听混淆. 比如: 有两个孩子让家长监听他们做作业.监听者是家长,被监听的对象是两个孩子.
参数:observer观察者,也就是KVO的订阅者,订阅者必须实现协议方法(下面有).keyPath描述将要观察的对象的属性,也就是被观察者的属性.optionsKVO的属性配置.NSKeyValueObservingOptionNewchange字典包括改变后的值NSKeyValueObservingOptionOldchange字典包括改变前的值NSKeyValueObservingOptionInitial注册后立刻触发KVO通知NSKeyValueObservingOptionPrior值改变前是否也要通知(这个key决定了是否在改变前改变后通知两次)
context上下文,这个会传递到协议方法中,用来区分消息,处理不同的KVO.所以应当是不同的.
- 解除监听
- (void)removeObserver:(NSObject *)observer forKeyPath:(NSString *)keyPath;删除指定keyPath的监听器.
- (void)removeObserver:(NSObject *)observer forKeyPath:(NSString *)keyPath context:(nullable void *)context NS_AVAILABLE(10_7, 5_0);删除特定上下文标记的指定keyPath的监听器. - 回调监听
- (void)observeValueForKeyPath:(nullable NSString *)keyPath ofObject:(nullable id)object change:(nullable NSDictionary<NSKeyValueChangeKey, id> *)change context:(nullable void *)context;
参数:keyPath被监听的keyPathobject被监听的修改后的对象,可以获取修改的对象的属性change保存信息改变的字典(可能有旧的值,新的值等context上下文
使用步骤
- 注册KVO监听.
- 实现代理方法.
- 移除监听.在dealloc方法中移除.
KVO使用注意事项
非常重要
- 当你在同一个ViewController中添加多个KVO的时候,无论哪个KVO都是走
- (void)observeValueForKeyPath:(nullable NSString *)keyPath ofObject:(nullable id)object change:(nullable NSDictionary<NSKeyValueChangeKey, id> *)change context:(nullable void *)context;回调方法.所以需要对想要的监听对象进行区分,以便指定不同的逻辑.
这里是对_tableView对象的contentOffset属性监听.
if (object == _tableView && [keyPath isEqualToString:@"contentOffset"]) {
[self doSomething]; }
}```
2. 我们假设当前类(在例子中为UITableViewController)还有父类,并且父类也有自己绑定了一些其他KVO呢?我们看到,上述回调函数体中只有一个判断,如果这个if不成立,这次KVO事件的触发就会到此中断了。但事实上,若当前类无法捕捉到这个KVO,那很有可能是在他的superClass,或者super-superClass...中,上述处理砍断了这个链。合理的处理方式应该是这样的:
```- (void)observeValueForKeyPath:(NSString *)keyPath ofObject:(id)object change:(NSDictionary *)change context:(void *)context
{
if (object == _tableView && [keyPath isEqualToString:@"contentOffset"]) {
[self doSomethingWhenContentOffsetChanges];
} else {
[super observeValueForKeyPath:keyPath ofObject:object change:change context:context]; }
}```
但是这个是要自己搞清楚,父类中到底有没有注册KVO.如果监听一个对象的两个属性,两个属性的改变时分开执行的,就会触发两次代理方法.如图:

3. KVO的一个特性,当对同一个`keyPath`进行多余一次的`removeObserver `的时候会导致程序crash.这种情况常常出现在父类有一个kvo,父类在dealloc中remove了一次,子类又remove了一次的情况下。不要以为这种情况很少出现!当你封装framework开源给别人用或者多人协作开发时是有可能出现的,而且这种crash很难发现.解决办法就是我们可以分别在父类以及本类中定义各自的context字符串,这样iOS就能知道移除的是自己的kvo,而不是父类中的kvo,避免二次remove造成crash.
4. 把监听到对象的属性值改变赋值的时候,一定要注意监听对象的值的类型.
把监听到对象的属性值改变赋值的时候,一定要注意监听对象的值的类型.
把监听到对象的属性值改变赋值的时候,一定要注意监听对象的值的类型.
**重要的事情说三遍**
5. 如果监听一个对象的多个属性,任何一个属性的改变都会走代理方法,也就是说对属性的监听,是分开执行的.
> 全部代码

1. MCBuyData.h
```#import <Foundation/Foundation.h>
@interface MCBuyData : NSObject
@property (nonatomic, assign) NSInteger number;
@property (nonatomic, assign) NSInteger money;
@end```
2. MCBuyData.m
```#import "MCBuyData.h"
@implementation MCBuyData
@end```
3. ViewController.h
```#import <UIKit/UIKit.h>
@class MCBuyData;
@interface ViewController : UIViewController
@property (nonatomic, strong) MCBuyData * buyData;
@end```
4. ViewController.m
``` #import "ViewController.h" ```
```#import "Masonry.h"```
```#import "MCBuyData.h"```
```#define kNumber @"number"```
```#define kMoney @"money"```
@interface ViewController ()
@property (nonatomic, strong) UILabel * numberLabel;
@property (nonatomic, strong) UILabel * moneyLabel;
@property (nonatomic, strong) UIButton * toBuyButton;
@end
@implementation ViewController
- (void)viewDidLoad {
[super viewDidLoad];
[self settingObserver];
[self initUI];
} - (void)dealloc {
[self.buyData removeObserver:self forKeyPath:kNumber context:@"number"];
[self.buyData removeObserver:self forKeyPath:kMoney context:@"money"];
}
pragma mark - 点击事件
- (void)toBuyButtonClicked {
NSInteger number = [[self.buyData valueForKey:kNumber] integerValue];
number += 1;
[self.buyData setValue:@(number) forKey:kNumber];
NSInteger money = [[self.buyData valueForKey:kMoney] integerValue];
money += 100;
[self.buyData setValue:@(money) forKey:kMoney];
}
-(void)observeValueForKeyPath:(NSString *)keyPath ofObject:(id)object change:(NSDictionary *)change context:(void *)context {
// 该change的内容记录的是本次监听到的属性的改变.
NSLog(@"change: %@",change);
NSString * new = change[@"new"];
if (object == self.buyData && [keyPath isEqualToString:kNumber] && (context == @"number")) {
self.numberLabel.text = [NSString stringWithFormat:@"次数: %@",new];
} else {
// 写了这句,如果父视图中没有注册的KVO,就会崩掉.
// reason: '<ViewController: 0x7fd7af406030>: An -observeValueForKeyPath:ofObject:change:context: message was received but not handled.
// [super observeValueForKeyPath:keyPath ofObject:object change:change context:context];
}
if ([keyPath isEqualToString:kMoney]) {
self.moneyLabel.text = [NSString stringWithFormat:@"金额: %@",new];
} else {
// [super observeValueForKeyPath:keyPath ofObject:object change:change context:context];
}
} - (void)settingObserver {
self.buyData = [[MCBuyData alloc] init];
[self.buyData setValue:@(0) forKey:kNumber];
[self.buyData setValue:@(0) forKey:kMoney];
[self.buyData addObserver:self forKeyPath:kNumber options:NSKeyValueObservingOptionNew | NSKeyValueObservingOptionOld context:@"number"];
[self.buyData addObserver:self forKeyPath:kMoney options:NSKeyValueObservingOptionNew | NSKeyValueObservingOptionOld context:@"money"];
} - (void)initUI {
[self.view addSubview:self.numberLabel];
[self.numberLabel mas_makeConstraints:^(MASConstraintMaker *make) {
make.left.mas_equalTo(self.view).with.offset(20);
make.right.mas_equalTo(self.view).with.offset(-20);
make.top.mas_equalTo(self.view).with.offset(100);
make.height.mas_equalTo(50);
}];
[self.view addSubview:self.moneyLabel];
[self.moneyLabel mas_makeConstraints:^(MASConstraintMaker *make) {
make.left.mas_equalTo(self.view).with.offset(20);
make.right.mas_equalTo(self.view).with.offset(-20);
make.top.mas_equalTo(self.view).with.offset(250);
make.height.mas_equalTo(50);
}];
[self.view addSubview:self.toBuyButton];
[self.toBuyButton mas_makeConstraints:^(MASConstraintMaker *make) {
make.left.mas_equalTo(self.view).with.offset(20);
make.right.mas_equalTo(self.view).with.offset(-20);
make.top.mas_equalTo(self.view).with.offset(400);
make.height.mas_equalTo(50);
}];
}
pragma mark - setter & getter
- (UILabel *)numberLabel {
if (_numberLabel == nil) {
self.numberLabel = [[UILabel alloc] init];
self.numberLabel.backgroundColor = [UIColor orangeColor];
self.numberLabel.font = [UIFont systemFontOfSize:15];
self.numberLabel.textColor = [UIColor whiteColor];
self.numberLabel.textAlignment = NSTextAlignmentCenter;
self.numberLabel.text = @"次数: 1";
} return _numberLabel;
} - (UILabel *)moneyLabel {
if (_moneyLabel == nil) {
self.moneyLabel = [[UILabel alloc] init];
self.moneyLabel.backgroundColor = [UIColor orangeColor];
self.moneyLabel.font = [UIFont systemFontOfSize:15];
self.moneyLabel.textColor = [UIColor whiteColor];
self.moneyLabel.textAlignment = NSTextAlignmentCenter;
self.moneyLabel.text = @"金额: 1";
} return _moneyLabel;
} - (UIButton *)toBuyButton {
if (_toBuyButton == nil) {
self.toBuyButton = [UIButton buttonWithType:UIButtonTypeCustom];
self.toBuyButton.titleLabel.font = [UIFont systemFontOfSize:14];
self.toBuyButton.backgroundColor = [UIColor redColor];
[self.toBuyButton setTitle:@"买 买 买!!!" forState:UIControlStateNormal];
[self.toBuyButton setTitleColor:[UIColor whiteColor] forState:UIControlStateNormal];
[self.toBuyButton addTarget:self action:@selector(toBuyButtonClicked) forControlEvents:UIControlEventTouchUpInside];
} return _toBuyButton;
}
@end
> Demo 下载地址
<https://github.com/mancongiOS/KVO.git>
> 说明
1. KVO注意事项1,2,3条转载于 编程小翁@博客园
详细的KVO总结,包括基本改变,使用案例,注意点.看我就够了!的更多相关文章
- KVO,看我就够了!
概述 KVO全称Key-Value-Observing,也叫键值监听,是一种观察者设计模式.提供了一种机制,当指定的对象的属性被修改后,对象就会收到一个通知.也就是说每次指定的被观察的对象的属性被修改 ...
- iOS—如何申请苹果公司开发者账号流程详细图文介绍(包括邓白氏编码的申请方法详细介绍)
我们要申请开发者账号,首先就需要先注册一个苹果的apple id,然后再这个账号的基础上去继续,这个相信大家都知道 这是申请appleid的地址:https://appleid.apple.com/a ...
- ajax详细讲解和封装包括HTTP状态码
AJAX(异步的JavaScript和XML,用异步的形式去操作xml) 主要的作用:数据交互 好处: 1.节省用户的操作时间 2.提高用户 ...
- Oracle11g在虚拟机win7上的详细安装过程(包括win7在虚拟机上的安装)
http://www.imsdn.cn/这个是镜像文件的下载地址,之前下载雨林和深度的VM识别不了. 这个好了之后就可以去这个网址下看安装教程很详细.https://blog.csdn.net/u01 ...
- Vue2全家桶之二:vue-router(路由)详细教程,看这个就够了
作者:东西里本文转载于:https://www.jianshu.com/p/514c7588e877来源:简书 转载仅供自己日后看方便. 由于Vue在开发时对路由支持的不足,于是官方补充了vue- ...
- Spring4.0+Hibernate4.0+Struts2.3整合包括增删改查案例,解决整合中出现的异常
源码下载:http://download.csdn.net/detail/cmcc_1234/7034775 ======================Application.xml======== ...
- 比较详细的mysql的几种连接功能分析,只要你看完就能学会的好东西
下面是例子分析表A记录如下: aID aNum 1 a20050111 2 a20050112 3 a20050113 4 ...
- ES 2021 来了,详细解读5个新特性,附案例
ES 2021是世界上最受欢迎的编程语言的最新版本〜 本次迭代中包含了五个新特性,让我们来一睹为快. 1.全部替换replaceAll: js默认的replace 方法仅替换字符串中一个模式的第一个实 ...
- 全网最详细的新手入门Mysql命令和基础,小白必看!
MySQL简介 什么是数据库 ? 数据库(Database)是按照数据结构来组织.存储和管理数据的仓库,它产生于距今六十多年前,随着信息技术和市场的发展,特别是二十世纪九十年代以后,数据管理不再仅仅是 ...
随机推荐
- OI中组合数的若干求法与CRT
OI中组合数的若干求法与CRT 只是下决心整理一下子呢~ 说明:本篇文章采用\(\binom{a}{b}\)而不是\(C_{a}^b\),以\(p\)指代模数,\(fac_i\)指代\(i!\),\( ...
- C&C++——基本说明
预处理器(Preprocessor) 1 . 用预处理指令#define 声明一个常数,用以表明1年中有多少秒(忽略闰年问题) #define SECONDS_PER_YEAR (60 * 60 * ...
- 接到新数据库时,分析业务常用的SQL语句
USE DataBaseName--清空当前GridView显示,释放内存: SELECT GETDATE() --数据库关系图 SELECT * FROM sysdiagrams --列出所有表 S ...
- 面试前需要弄懂的SQL
说明:创建数据库 view source print? 1 Create DATABASE database-name 说明:删除数据库 view source print? 1 drop d ...
- Codeforces Round #350 (Div. 2) D1
D1. Magic Powder - 1 time limit per test 1 second memory limit per test 256 megabytes input standard ...
- Vue2.0关于生命周期和钩子函数
Vue生命周期简介: Vue1.0+和Vue2.0在生命周期钩子上的区别还是很大的,如下: 代码验证: <!DOCTYPE html> <html> <head& ...
- html中offsetTop、clientTop、scrollTop、offsetTop各属性介绍(转载)
HTML精确定位:scrollLeft,scrollWidth,clientWidth,offsetWidth scrollHeight: 获取对象的滚动高度. scrollLeft: 设置或获取位于 ...
- js闭包,原型,作用域等再一次理解
要理解闭包,原型等,首先要理解作用域 作用域:就是函数在定义的时候创建的,用于寻找使用到的变量的值的一个索引,而他内部的规则是,把函数自身的本地变量放在最前面,把自身的父级函数中的变量放在其次,把再高 ...
- bzoj2002 弹飞绵羊 分块
这道题是分块的初尝试 讲给定的区间n进行分块处理 这个每次修改的复杂的只有logn 很方便 代码是学黄学长的 http://hzwer.com/3505.html 当然里面还是有一定我自己的想法在里面 ...
- CCCC练习即感
字符串进行初始化时不能通过char a[10]={'\0'}来简单进行,写循环或者memset,亲测有效,以及初始化分好情况,用空格还是'\0',别乱搞. 有一个有意思的题,连续因子,从2开始,依次向 ...