很久之前写了一份传值的文章- 传值
里面记录了方法传值,代理,通知以及Block传值。
不够全面,还有KVO没有介绍,在这里补上,同时比对一下各自的不同
KVO(key-value-observing键值观察)
把对象的某个属性作为键,当键的值改变时,就会唤起响应方法。实现KVO,要先添加监听
<#observer#>监听的对象
<#keyPath#>监听的对象属性
<#options#>监听的类型
<#context#>传入的上下文对象
添加了监听之后,我们需要响应监听
<#keyPath#>监听的属性
<#object#>监听的对象
<#change#>改变的值,从上面的注释来看,change里的值,跟设置监听时传入的option有关。
当你在设置监听时,option选择了<NSKeyValueObservingOptionNew>,此处就会返回新的值,当设置了<NSKeyValueObservingOptionOld>,此处返回改变前的值,当你两种都传入时,我们就可以拿到改变前的值跟改变后的值
<#context#>上下文对象,从上方注释来看,context对象永远是你注册监听时传入的相同上下文对象
上代码
@interface Student : NSObject
@property(nonatomic,copy)NSString *name;
@end
@implementation Student
- (void)observeValueForKeyPath:(NSString *)keyPath ofObject:(id)object change:(NSDictionary<NSKeyValueChangeKey,id> *)change context:(void *)context {
NSLog(@"%@",object);
}
- (void)dealloc {
[self removeObserver:self forKeyPath:@"name"];
}
@end
int main(int argc, const char * argv[]) {
@autoreleasepool {
// insert code here...
Student *stu = [Student new];
[stu addObserver:stu forKeyPath:@"name" options:NSKeyValueObservingOptionNew|NSKeyValueObservingOptionOld context:nil];
stu.name = @"asd";
}
return 0;
}
结果如下,如此,就完成了基本的监听,当然,别忘了移除监听
KVO的介绍到此为止。接下来比对一下各自的区别。
-
KVO:监听对象,对象属性变化时,接收信息,可以看到属性的变化过程
-
通知:在通知中心签署通知,通知中心根据签署的通知名称,发出消息到所有签署对应通知的对象上,一对多,不保证所有签署者都可以收到通知
-
代理:雇主不做事,由代理去处理响应方法。代理属性使用weak属性,不增加内存,但如果方法多的话,会有大量的方法实现代码
- 代理签署协议
- 判断是否实现协议方法
- 实现协议方法
-
block:类似于方法函数,使用copy来定义,会引起内存增加,需要注意循环引用问题.