聊聊 KVO 与 KVC 吧

先聊聊 KVO 与 KVC 的区别吧:
KVO是指键-值-观察者模式, 键值监听, 监听一个对象属性值的改变。KVO是基于KVC的。
KVC 是指键-值编码, 通过一个字符串的 key 来找到value , 是 value for key 方法, 直接或通过实例变量访问的机制 。利用 KVC 可以随意修改一个对象的属性或者成员变量, 并且私有变量也可修改。

一. KVO

KVO是指键-值-观察者(key-value-observe),
是一种使对象获取其他对象的特定属性变化的通知机制。
与NSNotification 不同的是。KVO 不需要通知中心对象。而是在对象属性变化之后会直接通知观察者。

KVO的步骤:

**1. 注册观察者 **
为了正确接收属性的变化通知,观察者对象必须先发一个消息给被观察者对象

- (void)addObserver:(NSObject *)observer forKeyPath:(NSString *)keyPath options:(NSKeyValueObservingOptions)options context:(nullable void *)context;

/*
NSKeyValueObservingOptions 可选的是一个枚举值。我们通常用到的是两个
NSKeyValueObservingOptionOld | NSKeyValueObservingOptionNew
当属性发生变化时。我们可以把旧值和新值传递给观察者 
*/

**2.接收变化通知 **
应该注意的是如果只是使用成员变量改变值的话是不会触发KVO的。要使用点语法,或者是KVC的方式改变值

// object 是被监听对象
- (void)observeValueForKeyPath:(NSString *)keyPath
                      ofObject:(id)object
                        change:(NSDictionary *)change
                       context:(void *)context;

**3.移除观察者身份 **
在不需要观察时要进行移除

- (void)removeObserver:(NSObject *)observer
            forKeyPath:(NSString *)keyPath;

二.KVC

什么是KVC
KVC是指key-value-code,键-值编码,是一种用于间接访问对象属性的机制。使用KVC可以直接修改对象属性,即使是私有的也可以访问。 如果是基本数据类型的应该封装一下。

KVC的基本使用有下面几点:
键值访问
路径访问
取数组内的数据
一些简单的运算
下面按照这几点用法来介绍一下

// 为了方便以后操作,我们先简单定义一下几个类

// Person类
#import 
#import "Totoro.h"

@interface Person : NSObject

@property (nonatomic, copy) NSString *name;
@property (nonatomic, assign) NSUInteger age;
@property (nonatomic, strong) Totoro *Totoro;
@property (nonatomic, strong) NSArray *Totoros;

@end

// Totoro类
#import 

@interface Totoro : NSObject

@property (nonatomic, assign) NSUInteger weight;

@end

// 为了证明上面说的可以修改私有属性,我们为 Totoro 添加一个私有的属性
#import "Totoro.h"

@interface Totoro ()

@property (nonatomic, copy) NSString *name;

@end

@implementation Totoro

@end

基本设置完了之后,我们可以在Main函数里面进行一下操作

// 先创建一个人对象
Person *me = [[Person alloc] init];
Totoro *Totoro = [[Totoro alloc] init];
me.Totoro = Totoro; // 我养了一只龙猫

按照我们最为传统的赋值方法我们要给人赋值一个name的话 我们通常会使用点语法进行赋值

// 本质上是调用了 [me setName:@"smile丽"];
me.name = @"smile丽"; // 这里相当于调用了setter方法

// 本质上是调用了 NSLog(@"%@", [me name]);
NSLog(@"%@", me.name); // 这里相当于调用getter方法

1> 键值访问

那我们来看一下 使用KVC的方式应该如何去赋值.使用KVC, 会自动寻找成员变量(xxx),如果找不到,然后再去掉去寻找,如果再找不到,就会报错。而不是去调用setter 和 getter 方法

// @() 对基本数据类型封装成对象
 [me setValue:@(24) forKey:@"age"];
 NSLog(@"%@", [me valueForKey:@"age"]);

2> 路径访问

什么是路径访问,对于一个类来说,可能他的属性是其他的类,如果要修改这里的属性。我们需要先通过路径来寻找到该属性,然后再进行赋值.

[me setValue:@"大白" forKeyPath:@"totoro.name"]; // 注意这里的(.)只是路径不是点语法
NSLog(@"%@", [me valueForKeyPath:@"totoro.name"]);

3> 取数组的数据

对于如果我们的数组里面存放的是对象,如果我们要获取数组里面每个对象的属性。这样的话,最容易的方法就是遍历数组。然后取出每个属性进行添加到数组中。这时候我们也可以使用KVC快速解决这种问题.
为了测试,我给Person一个totoros的数组属性,下面造一下数据.

NSMutableArray *totoros = [NSMutableArray array];
for (int i = 0; i < 4; i++) {
    Totoro *totoro = [[totoro alloc] init];
    NSString *name = [NSString stringWithFormat:@"大白_%d", i];
    [totoro setValue:name forKey:@"name"];
    NSUInteger weight = 3.8 + i;
    [totoro setValue:@(weight) forKey:@"weight"]; // 这个数据在第四点用到
    [totoros addObject:totoro];
}

我们如何实现上述的需求呢

使用 KVC回去属性的数组

NSMutableArray *array = [totoros mutableArrayValueForKeyPath:@"name"];
NSLog(@"%@", array);
/*
 (
 "大白_0",
 "大白_1",
 "大白_2",
 "大白_3"
 )
*/

4> 一些简单的运算
可以使用的关键字: 数量@count, 最大值@max, 最小值@min, 和@sum

me.totoros = totoros;
// 取到所有的相关属性元素,进行计算, 由于方法返回的是 id, 所以要使用对象接收,我们可以使用 NSNumber, 而不是 int 之类的
NSLog(@"数量:%@", [p valueForKeyPath:@"totoros.@count"]);
NSLog(@"平均体重%@", [p valueForKeyPath:@"[email protected]"]);

你可能感兴趣的:(聊聊 KVO 与 KVC 吧)