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
描述将要观察的对象的属性,也就是被观察者的属性.options
KVO的属性配置.NSKeyValueObservingOptionNew
change字典包括改变后的值NSKeyValueObservingOptionOld
change字典包括改变前的值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
属性监听.
- (void)observeValueForKeyPath:(NSString *)keyPath ofObject:(id)object change:(NSDictionary *)change context:(void *)context { if (object == _tableView && [keyPath isEqualToString:@"contentOffset"]) { [self doSomething]; } }
- 我们假设当前类(在例子中为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.如果监听一个对象的两个属性,两个属性的改变时分开执行的,就会触发两次代理方法.如图:
- KVO的一个特性,当对同一个
keyPath
进行多余一次的removeObserver
的时候会导致程序crash.这种情况常常出现在父类有一个kvo,父类在dealloc中remove了一次,子类又remove了一次的情况下。不要以为这种情况很少出现!当你封装framework开源给别人用或者多人协作开发时是有可能出现的,而且这种crash很难发现.解决办法就是我们可以分别在父类以及本类中定义各自的context字符串,这样iOS就能知道移除的是自己的kvo,而不是父类中的kvo,避免二次remove造成crash. - 把监听到对象的属性值改变赋值的时候,一定要注意监听对象的值的类型.
把监听到对象的属性值改变赋值的时候,一定要注意监听对象的值的类型.
把监听到对象的属性值改变赋值的时候,一定要注意监听对象的值的类型.
重要的事情说三遍 - 如果监听一个对象的多个属性,任何一个属性的改变都会走代理方法,也就是说对属性的监听,是分开执行的.
全部代码
- MCBuyData.h
#import <Foundation/Foundation.h> @interface MCBuyData : NSObject @property (nonatomic, assign) NSInteger number; @property (nonatomic, assign) NSInteger money; @end
- MCBuyData.m
#import "MCBuyData.h" @implementation MCBuyData @end
- ViewController.h
#import <UIKit/UIKit.h> @class MCBuyData; @interface ViewController : UIViewController @property (nonatomic, strong) MCBuyData * buyData; @end
- 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: ' - (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
说明
- KVO注意事项1,2,3条转载于 编程小翁@博客园
KVO,看我就够了!的更多相关文章
- 详细的KVO总结,包括基本改变,使用案例,注意点.看我就够了!
概述 KVO全称Key-Value-Observing,也叫键值监听,是一种观察者设计模式.提供了一种机制,当指定的对象的属性被修改后,对象就会收到一个通知.也就是说每次指定的被观察的对象的属性被修改 ...
- 【原】HTTP in iOS你看我就够
声明:本文是本人 编程小翁 原创,转载请注明. 本文同步发布在简书中,强烈建议移步简书查看,编程小翁 HTTP属于老话题了,在项目中我们经常需要往服务端发POST或者GET请求,但是对于HTTP的了解 ...
- HTTP in iOS你看我就够
HTTP属于老话题了,在项目中我们经常需要往服务端发POST或者GET请求,但是对于HTTP的了解不应只局限于此.千里之行,始于足下.越想走的远,基本原理就应该了解的透彻全面一些,仅仅停留在使用ASI ...
- 关于iOS多线程,你看我就够了
在这篇文章中,我将为你整理一下 iOS 开发中几种多线程方案,以及其使用方法和注意事项.当然也会给出几种多线程的案例,在实际使用中感受它们的区别.还有一点需要说明的是,这篇文章将会使 用 Swift ...
- Prometheus Metrics 设计的最佳实践和应用实例,看这篇够了!
Prometheus 是一个开源的监控解决方案,部署简单易使用,难点在于如何设计符合特定需求的 Metrics 去全面高效地反映系统实时状态,以助力故障问题的发现与定位.本文即基于最佳实践的 Metr ...
- 修改版: 小伙,多线程(GCD)看我就够了,骗你没好处!
多线程(英语:multithreading),是指从软件或者硬件上实现多个线程并发执行的技术.具有多线程能力的计算机因有硬件支持而能够在同一时间执行多于一个线程,进而提升整体处理性能.具有这种能力的系 ...
- html总结----------------------看这个就够了
HTML是我们学习Javaweb的第一步 很好地掌握门课是非常有必要的!下面就是我在听资深老师讲课的笔记!个人觉得非常不错!希望可以帮助到那些在学习javaweb路上的 朋友们!从今天 陆续的整理这门 ...
- 做一款仿映客的直播App?看我就够了
来源:JIAAIR 链接:http://www.jianshu.com/p/5b1341e97757 一.直播现状简介 1.技术实现层面: 技术相对都比较成熟,设备也都支持硬编码.IOS还提供现成 ...
- Vue2全家桶之二:vue-router(路由)详细教程,看这个就够了
作者:东西里本文转载于:https://www.jianshu.com/p/514c7588e877来源:简书 转载仅供自己日后看方便. 由于Vue在开发时对路由支持的不足,于是官方补充了vue- ...
随机推荐
- Init.rc分析(刘举奎)
http://www.360doc.com/content/14/0926/20/13253385_412582822.shtml
- markdown表格
markdown制作表格 一. 使用原生html表格标签制作 <table> <tr> <td>表头</td> </tr> <tr&g ...
- sublime text 我的常用配置
{ "color_scheme": "Packages/Color Scheme - Default/IDLE.tmTheme", "font_fac ...
- iOS开发者需要的5款排版工具
Attributed String Creator Attributed String Creator可以从你的格式化文本中自动生成原生的Objective-C代码.你可以将文本写入.粘贴或者导入At ...
- HDU 5620 KK's Steel
想了一下发现是斐波那契数列.....水题 #include <stdio.h> #include <algorithm> #include <string.h> # ...
- mysql数据库中间件研究
随着互联网的发展,数据量的不断增大. 单台实例已经远远无法满足业务的需要. 对数据库分库分表的需求不断的增加随之而来的就是数据库中间件的开发. 一. 单台实例主要面临下面几个问题: 1. 数据量太大 ...
- SPOJ839 OPTM - Optimal Marks
传送门 闵神讲网络流应用的例题,来水一水 要写出这道题,需要深入理解两个概念,异或和最小割. 异或具有相对独立性,所以我们把每一位拆开来看,即做大概$32$次最小割.然后累加即可. 然后是最小割把一张 ...
- CodeSmith生成实体类
1.新建一个模板,将以下代码复制进去,在属性工具栏里设置 SourceTtable和NameSpace , 点击生成即可 <%@ CodeTemplate Language="C#&q ...
- C语言-基础
程序是为了让计算机完成某项任务而编写的逐条执行的指令序列. C语言的特点是:结构化,语言简洁,功能强大,移植性好. 因为移植性好,所以大多数单片机如51.stm32,msp430,等众多嵌入式微处理器 ...
- jade的特点
jade特点 1超强的可读性 2灵活易用的缩进 3块扩展 4代码默认进过编码处理,以增强安全性 5编译及运行时的上下文错误报告 6命令行编译支持 7html5模式(使用 !!!5文档类型) 8可选的内 ...