详细的KVO总结,包括基本改变,使用案例,注意点.看我就够了!

概述

  1. KVO全称Key-Value-Observing,也叫键值监听,是一种观察者设计模式.提供了一种机制,当指定的对象的属性被修改后,对象就会收到一个通知.也就是说每次指定的被观察的对象的属性被修改后,KVO就会自动通知相应的观察者.
  2. 优势:可以降低两个类(业务逻辑和视图控制的类)之间的耦合性.也就是说可以很容易的实现视图组件和数据模型的分离.当数据模型的属性值改变之后作为监听器的视图组件就会被激发,激发时就会回调监听器自身.
  3. 在Objective-C中要实现KVO则必须实现NSKeyValueObServing协议.但不用担心,因为NSObject已经实现了该协议,因此几乎所有的Objective-C对象都可以使用KVO.

KVO的方法

  1. 监听方法
    - (void)addObserver:(NSObject *)observer forKeyPath:(NSString *)keyPath options:(NSKeyValueObservingOptions)options context:(nullable void *)context;
    某一个对象(受虐狂,喜欢被人监视),给自己添加一个监听者(一般都是控制器本身self),让监听者监听自身的某一个属性. options就是要求监听者记录的信息. context就是要监听者给自己添加一个标记,以防止和别的对象的监听混淆. 比如: 有两个孩子让家长监听他们做作业.监听者是家长,被监听的对象是两个孩子.
    参数:
  • observer 观察者,也就是KVO的订阅者,订阅者必须实现协议方法(下面有).
  • keyPath 描述将要观察的对象的属性,也就是被观察者的属性.
  • options KVO的属性配置.
    • NSKeyValueObservingOptionNewchange字典包括改变后的值
    • NSKeyValueObservingOptionOldchange字典包括改变前的值
    • NSKeyValueObservingOptionInitial注册后立刻触发KVO通知
    • NSKeyValueObservingOptionPrior值改变前是否也要通知(这个key决定了是否在改变前改变后通知两次)
  • context上下文,这个会传递到协议方法中,用来区分消息,处理不同的KVO.所以应当是不同的.
  1. 解除监听
    - (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的监听器.
  2. 回调监听
    - (void)observeValueForKeyPath:(nullable NSString *)keyPath ofObject:(nullable id)object change:(nullable NSDictionary<NSKeyValueChangeKey, id> *)change context:(nullable void *)context;
    参数:
    • keyPath 被监听的keyPath
    • object 被监听的修改后的对象,可以获取修改的对象的属性
    • change 保存信息改变的字典(可能有旧的值,新的值等
    • context 上下文

使用步骤

  1. 注册KVO监听.
  2. 实现代理方法.
  3. 移除监听.在dealloc方法中移除.

KVO使用注意事项

非常重要

  1. 当你在同一个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]; } }
  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.如果监听一个对象的两个属性,两个属性的改变时分开执行的,就会触发两次代理方法.如图:

1.png

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

全部代码

2.png

  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条转载于 编程小翁@博客园
  • 0
    点赞
  • 1
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
《Tableau超详细教程-案例实践项目.pdf》是一本关于Tableau数据可视化工具的详细教程,该教程旨在帮助读者通过实践项目的方式,从入门到精通掌握Tableau的使用方法和技巧。 这本教程包含了从基础知识到高级技巧的全面内容。首先,教程会逐步介绍Tableau的界面和基本操作,以及数据连接和导入的方法。接着,教程会详细介绍Tableau的数据处理和准备功能,包括数据清洗、数据分割、合并和转换等。读者将通过实践项目了解如何有效地处理和准备数据,以便在Tableau中进行可视化。 教程还会引导读者学习Tableau的可视化功能,包括常用的图表类型(如条形图、折线图、饼图等)和交互式的仪表板设计。通过实践项目,读者将了解如何选择和设计最合适的图表类型来呈现数据,并学习如何添加过滤器、参数和动画效果来提升可视化效果。 此外,教程还将介绍Tableau的高级功能和技巧,如计算字段、表计算、集成分析和地理空间分析等。读者将通过实践项目深入了解这些高级功能,并学会如何利用它们来进行更复杂和深入的数据分析和可视化。 总之,《Tableau超详细教程-案例实践项目.pdf》是一本全面而详细的教程,通过实践项目的方式帮助读者从零基础到精通掌握Tableau的使用方法和技巧。无论是初学者还是有一定经验的用户,都可以通过这本教程提升自己在数据可视化领域的技能水平。

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值