美文网首页
iOS开发之基础篇(15)—— KVC、KVO

iOS开发之基础篇(15)—— KVC、KVO

作者: 看影成痴 | 来源:发表于2017-11-09 18:02 被阅读34次

版本

Xcode 9.1

KVC

1、概述

KVC(Key Value Coding)即键值编码,能简便地动态读写对象属性,其实现方法是使用字符串来描述需要更改的对象属性。
KVC的操作方法由NSKeyValueCoding协议提供,而NSObject遵循了该协议,所以说,OC中几乎所有的对象都支持KVC操作。

2、操作方法

  • 写入操作
    setValue:(nullable id) forKey:(NSString *) 用于简单路径
    setValue:(nullable id) forKeyPath:(NSString *) 用于复合路径
  • 读取操作
    valueForKey:(NSString *) 用于简单路径
    valueForKeyPath:(NSString *) 用于复合路径

所谓简单路径,指访问对象本身的属性;所谓复合路径,指访问对象属性里的对象的属性(比如对象A属性里面包含对象B,对象B有属性name,那么对象A访问对象B的name属性即为复合路径)。

示例:
对象Person里面包含属性name、age及对象Dog。对象Dog里有属性name、age。

Person.h

#import "Dog.h"

@interface Person : NSObject

@property (nonatomic, copy)     NSString    *name;
@property (nonatomic, assign)   NSInteger   age;

@property (nonatomic, retain)   Dog         *dog;

@end

Dog.h

@interface Dog : NSObject

@property (nonatomic, copy)     NSString    *name;
@property (nonatomic, assign)   NSInteger   age;

@end

main.m

    // 实例化一个Person
    Person *person = [[Person alloc] init];
    
    // 简单路径的写入操作
    [person setValue:@"King" forKey:@"name"];
    [person setValue:@23 forKey:@"age"];        // 注

    // 简单路径的写入操作(用NSDictionary批量设置)
//    NSDictionary *dic = @{@"name":@"King", @"age":@23,};
//    [person setValuesForKeysWithDictionary:dic];
    
    // 简单路径的读取操作
    NSLog(@"Person name = %@", [person valueForKey:@"name"]);
    NSLog(@"Person age = %ld", [[person valueForKey:@"age"] integerValue]);
    
    
    // 实例化person里的Dog对象属性,否则为nil不能进行如下操作
    person.dog = [[Dog alloc] init];
    
    // 复杂路径的写入操作
    [person setValue:@"XiaoHei" forKeyPath:@"dog.name"];
    [person setValue:@3 forKeyPath:@"dog.age"];
    
    // 复杂路径的读取操作
    NSLog(@"Dog name = %@", [person valueForKeyPath:@"dog.name"]);
    NSLog(@"Dog age = %ld", [[person valueForKeyPath:@"dog.age"] integerValue]);

注:@3是一种简便写法,相当于[NSNumber numberWithInt:3];又如@[]代表数组,@{}代表NSDictionary。如果直接将一个int赋值给id类型的数据,编译会报错。

输出结果:

3、底层实现

  • 写入操作时(例如setValue: forKey:@"A"),方法内部会做以下操作:
  1. 检查是否存在相应key的setter方法(setA),如存在则调用setter方法;
  2. 如果没有setter方法,就会查找与key相同名称并且带下划线的成员变量(_A),如果有则直接赋值;
  3. 如果没有带下划线的成员变量,则搜索与key同名的成员变量(A),有则直接赋值;
  4. 如果最后仍没找到,则调用setValue: forUndefinedKey:方法。
  • 读取操作时(例如valueForKey:@"A"),方法内部会做以下操作:
  1. 检查是否存在相应key的getter方法(A),如存在则调用getter方法;
  2. 如果没有getter方法,就会查找与key相同名称并且带下划线的成员变量(_A),如果有则直接读取;
  3. 如果没有带下划线的成员变量,则搜索与key同名的成员变量(A),有则直接读取;
  4. 如果最后仍没找到,则调用valueForUndefinedKey:方法。

setValue: forUndefinedKey:和valueForUndefinedKey:方法默认实现都是抛出异常,我们可以根据需要重写它们。

KVO

1、概述

KVO(Key Value Observing)即键值监听,是一种观察者模式,通过对某个对象的某个属性添加监听,当该属性改变时,会调用相应方法。
KVO的操作方法由NSKeyValueObserving协议提供,而NSObject遵循了该协议,所以说,OC中几乎所有的对象都支持KVO操作。

2、操作方法

  • 注册指定key路径的观察者:addObserver: forKeyPath: options: context:
  • 回调监听:observeValueForKeyPath: ofObject: change: context:
  • 删除指定key路径的观察者:removeObserver: forKeyPath:

示例(由KVC示例进化):

#import "ViewController.h"
#import "Person.h"

@interface ViewController () {
    
    Person *person;
}

@end

@implementation ViewController

- (void)viewDidLoad {
    [super viewDidLoad];
    
    // 实例化一个Person
    person = [[Person alloc] init];     // 因为要移除监听,所以把person设为全局变量
    
    // 简单路径的写入操作
    [person setValue:@"King" forKey:@"name"];
    [person setValue:@23 forKey:@"age"];        // 注
    
    // 简单路径的读取操作
    NSLog(@"Person name = %@", [person valueForKey:@"name"]);
    NSLog(@"Person age = %ld", [[person valueForKey:@"age"] integerValue]);
    
    
    // 实例化person里的Dog对象属性,否则为nil不能进行如下操作
    person.dog = [[Dog alloc] init];
    
    // 复杂路径的写入操作
    [person setValue:@"XiaoHei" forKeyPath:@"dog.name"];
    [person setValue:@3 forKeyPath:@"dog.age"];
    
    // 复杂路径的读取操作
    NSLog(@"Dog name = %@", [person valueForKeyPath:@"dog.name"]);
    NSLog(@"Dog age = %ld", [[person valueForKeyPath:@"dog.age"] integerValue]);
    
    /* 添加对name的监听(注册观察者)*/
    //第一个参数 observer:观察者 
    //第二个参数 keyPath: 被观察的属性名称
    //第三个参数 options: 观察属性的新值、旧值等的一些配置(枚举值,可以根据需要设置,例如这里可以使用两项)
    //第四个参数 context: 上下文,可以为 KVO 的回调方法传值(例如设定为一个放置数据的字典)
    [person addObserver:self forKeyPath:@"name" options:NSKeyValueObservingOptionNew|NSKeyValueObservingOptionOld context:nil];
    
    person.name = @"Haha";      // 这句会调用如下变化方法
}


/**
 当name发生变化时调用此方法
 
 @param keyPath 属性名称
 @param object 被观察的对象
 @param change 变化前后的值都存储在 change 字典中
 @param context 注册观察者时,context 传过来的值
 */
- (void)observeValueForKeyPath:(NSString *)keyPath ofObject:(id)object change:(NSDictionary<NSKeyValueChangeKey,id> *)change context:(void *)context {
    
    if ([keyPath isEqualToString:@"name"]) {
        NSLog(@"newValue:%@, oldValue:%@", [change objectForKey:@"new"],change[@"old"]);
    }
}


- (void)dealloc {
    
    // 移除监听
    [person removeObserver:self forKeyPath:@"name"];
}


@end

结果:

3、优缺点

  • 优点
  1. 能够提供一种简单的方法实现两个对象间的同步。
  2. 能够对系统对象的状态改变作出响应,而不需要改变内部对象的实现;
  3. 能够提供观察的属性的最新值以及先前值;
  4. 用key paths来观察属性,因此也可以观察嵌套对象;
  5. 监听对象可以是空的,为了防止意外崩溃;
  6. 完成了对观察对象的抽象,因为不需要额外的代码来允许观察值能够被观察。
  • 缺点
  1. 观察的属性(对应key)是字符串类型,纯手打容易出错,且编译器不会警告以及检查;
  2. 对属性重构将导致我们的观察代码不再可用;
  3. 只能监测对象属性,不能对方法或者动作做出反应。

相关文章

  • KVC

    iOS 如何使用KVC iOS开发UI篇—Kvc简单介绍 iOS开发系列--Objective-C之KVC、KVO

  • KVC 和 KVO

    iOS-KVC和KVO精炼讲解(干货)KVC 和 KVOiOS开发系列--Objective-C之KVC、KVO细...

  • iOS开发之基础篇(15)—— KVC、KVO

    版本 Xcode 9.1 KVC 1、概述 KVC(Key Value Coding)即键值编码,能简便地动态读写...

  • KVC、KVO

    IOS开发系列--Objective-C之KVC、KVO - KenshinCui - 博客园

  • ReactiveCocoa相关

    随手记录: 1、iOS开发系列--Objective-C之KVC、KVO 2、ReactiveCocoa初探

  • KVC

    KVC原理剖析 - CocoaChina_让移动开发更简单 iOS开发底层细究:KVC和KVO底层原理 | iOS...

  • iOS日记15-KVC

    1.iOS开发技巧系列---详解KVC 2.漫谈 KVC 与 KVO 3.KVC/KVO原理详解及编程指南 关键点...

  • iOS-KVO浅谈

    上一篇:iOS-KVC浅谈 前言:KVO 作为 KVC 的同袍兄弟,功能更强大,聊聊 KVO。 一、KVO 简介 ...

  • 知识点3

    26. 什么是KVC和KVO? KVO: iOS开发-KVO的奥秘 http://www.jianshu.com/...

  • KVC剖析与使用

    KVC与KVO相关面试题 KVC -- Key Value Coding(键值编码) 在iOS开发过程中,允许开发...

网友评论

      本文标题:iOS开发之基础篇(15)—— KVC、KVO

      本文链接:https://www.haomeiwen.com/subject/vgwpmxtx.html