先聊聊 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<NSString *,id> *)change
context:(void *)context;
**3.移除观察者身份 **
在不需要观察时要进行移除
- (void)removeObserver:(NSObject *)observer
forKeyPath:(NSString *)keyPath;
二.KVC
什么是KVC
KVC是指key-value-code,键-值编码,是一种用于间接访问对象属性的机制。使用KVC可以直接修改对象属性,即使是私有的也可以访问。 如果是基本数据类型的应该封装一下。
KVC的基本使用有下面几点:
键值访问
路径访问
取数组内的数据
一些简单的运算
下面按照这几点用法来介绍一下
// 为了方便以后操作,我们先简单定义一下几个类
// Person类
#import <Foundation/Foundation.h>
#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 <Foundation/Foundation.h>
@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:@"totoros.@avg.weight"]);