Object C KVC & KVO & NOtificati
2021-08-10 本文已影响0人
Yison_a169
一. KVC
间接的修改或获取对象的属性,降低程序(类与类)之间的耦合度.
常见的两种用法
1.对私有变量进行赋值,同样的也可以通过它进行取值
在使用KVC时, 一定要保证键值是存在的.
#KVC最为重要的四个方法
- (nullable id)valueForKey:(NSString *)key; //直接通过Key来取值
- (void)setValue:(nullable id)value forKey:(NSString *)key; //通过Key来设值
- (nullable id)valueForKeyPath:(NSString *)keyPath; //通过KeyPath来取值
- (void)setValue:(nullable id)value forKeyPath:(NSString *)keyPath; //通过KeyPath来设值属性
#NSKeyValueCoding类别中其他的一些方法:
//默认返回YES,表示如果没有找到Set<Key>方法的话,会按照_key,_iskey,key,iskey的顺序搜索成员,设置成NO就不这样搜索
+ (BOOL)accessInstanceVariablesDirectly;
//KVC提供属性值正确性验证的API,它可以用来检查set的值是否正确、为不正确的值做一个替换值或者拒绝设置新值并返回错误原因。
- (BOOL)validateValue:(inout id __nullable * __nonnull)ioValue forKey:(NSString *)inKey error:(out NSError **)outError;
//这是集合操作的API,里面还有一系列这样的API,如果属性是一个NSMutableArray,那么可以用这个方法来返回。
- (NSMutableArray *)mutableArrayValueForKey:(NSString *)key;
//如果Key不存在,且没有KVC无法搜索到任何和Key有关的字段或者属性,则会调用这个方法,默认是抛出异常。
- (nullable id)valueForUndefinedKey:(NSString *)key;
//和上一个方法一样,但这个方法是设值。
- (void)setValue:(nullable id)value forUndefinedKey:(NSString *)key;
//如果你在SetValue方法时面给Value传nil,则会调用这个方法
- (void)setNilValueForKey:(NSString *)key;
//输入一组key,返回该组key对应的Value,再转成字典返回,用于将Model转到字典。
- (NSDictionary<NSString *, id> *)dictionaryWithValuesForKeys:(NSArray<NSString *> *)keys;
例子
#import <Foundation/Foundation.h>
@interface Test: NSObject {
NSString *_name;
}
@end
@implementation Test
@end
int main(int argc, const char * argv[]) {
@autoreleasepool {
// insert code here...
//生成对象
Test *obj = [[Test alloc] init];
//通过KVC赋值name
[obj setValue:@"xiaoming" forKey:@"name"];
//通过KVC取值name打印
NSLog(@"obj的名字是%@", [obj valueForKey:@"name"]);
}
return 0;
}
2.字典转模型
obj.name=dic[@"name"];
obj.sex=dic[@"sex"];
obj.age=dic[@"age"];
//将以上代码替换为一行代码
[obj setValuesForKeysWithDictionary:dic];
二. KVO
对目标对象的某属性添加观察,当该属性发生变化时,通过触发观察者对象实现的KVO接口方法,来自动的通知观察者
*KVO使用步骤:
/***
keyPath:描述将要观察的属性,相对于被观察者。
options:KVO的一些属性配置;有四个选项。
NSKeyValueObservingOptionNew:change字典包括改变后的值
NSKeyValueObservingOptionOld:change字典包括改变前的值
NSKeyValueObservingOptionInitial:注册后立刻触发KVO通知
NSKeyValueObservingOptionPrior:值改变前是否也要通知(这个key决定了是否在改变前改变后通知两次)
context: 上下文,这个会传递到订阅着的函数中,用来区分消息,所以应当是不同的。
***/
[stuaddObserver:selfforKeyPath:@"name"options:NSKeyValueObservingOptionNew|NSKeyValueObservingOptionOld context:@"myObserver"];
//当指定键值发生改变时, 会自动调用这个方法
observeValueForKeyPath.
//所有观察者模式的性能都不好,需要及时移除。
removeObserver.
例子
//
// XNViewController.m
// KVC----KVO
//
// Created by neng on 14-6-21.
// Copyright (c) 2014年 neng. All rights reserved.
//
#import "XNViewController.h"
#import "XNPerson.h"
#import "XNStudent.h"
#import "XNBook.h"
@interface XNViewController ()
@end
/**
* KVC: Key Value Coding(键值编码)
* 间接修改/获取对象的属性, 降低类与类之间的耦合度.
* KVO: Key Value Observer(键值观察)
KVO通常用于观察”对象的某个属性“发生变化时,及时做出响应!
而NSNotificationCenter是需要POST"通知字符串"(表示监听的事件类型)的对象存在,通知中心才能够工作!
*/
@implementation XNViewController
- (void)viewDidLoad {
[super viewDidLoad];
//1.简单的修改对象属性
[self kvcDemo1];
//2.对于子类, 也能直接修改
[self kvcDemo2];
//KVO 演示
[self kvoDemo];
}
- (void)kvoDemo {
XNStudent *stu = [[XNStudent alloc] init];
stu.name = @"xuneng";
//设置监听对象
//1》 负责观察的对象 self
//2》 观察的键值路径
//3》 观察的选项
//4》 上下文:通常用于区分不同的观察事件
[stu addObserver:self forKeyPath:@"name" options:NSKeyValueObservingOptionNew | NSKeyValueObservingOptionOld context:@"myObserver"];
stu.name = @"neng"; //修改
//所有观察者模式的性能都不好, 需要及时移除
[stu removeObserver:self forKeyPath:@"name"];
stu.name = @"xu"; //移除后再次修改
}
/**
当KVO指定的对象键值发生改变时, 会自动调用该方法
name: 观察的键值
object: 观察的对象
change: 修改的字典(新旧数值)
context: 指定观察者时传入的上下文
*/
- (void)observeValueForKeyPath:(NSString *)keyPath ofObject:(id)object change:(NSDictionary *)change context:(voidvoid *)context {
NSLog(@"|-- %@ --|-- %@ --|-- %@ --|-- %@--|", keyPath, object, change, context);
}
/**
*简单的修改对象属性
*/
- (void)kvcDemo1 {
//1.原始的方式. 点语法修改对象属性的值
XNPerson *p1 = [[XNPerson alloc] init];
p1.name = @"xuneng";
p1.age = 24;
// NSLog(@"%@ , %d",p1.name,p1.age);
NSLog(@"KVC1 demo1--> %@", p1); //这样直接调用description方法.
//2.KVC来修改
[p1 setValue:@"xxxx" forKeyPath:@"name"];
[p1 setValue:@"10" forKeyPath:@"age"];
NSLog(@"KVC2 demo1--> %@ , %d", p1.name, p1.age);
}
/**
*对于子类, 也能直接改
*/
- (void)kvcDemo2 {
//1.传统方法
XNStudent *p1 = [[XNStudent alloc] init]; //学生
p1.name = @"student xuneng";
p1.age = 22;
// NSLog(@"%@ , %d",p1.name,p1.age);
NSLog(@"KVC1 demo2--> %@", p1); //这样直接调用description方法.
//2.KVC来修改子类
[p1 setValue:@"xxxx" forKeyPath:@"name"];
[p1 setValue:@"10" forKeyPath:@"age"];
NSLog(@"KVC2 demo2--> %@ , %d", p1.name, p1.age);
}
@end
1. 如何手动触发KVO?
答: 被监听的属性的值被修改时,就会自动触发KVO。如果想要手动触发KVO,则需要我们自己调用willChangeValueForKey和didChangeValueForKey方法即可在不改变属性值的情况下手动触发KVO,并且这两个方法缺一不可。
2. 直接修改成员变量会触发KVO吗?
答:不会触发
3. 通过KVC修改属性会触发KVO么?
答:会触发
三、NSNotification
使用观察者模式来实现的用于跨层传递消息。往往也用NSNotification来实现解耦的目的。
常见的还有代理传值、block传值等。(代理实现和block一般用于一对一的情况。)