首先我们创建两个类一个Person类和一个Teacher类,在Teacher类中有个name属性
#import <Foundation/Foundation.h>
@interface Teacher : NSObject
/**<#注释#>*/
@property (nonatomic , strong) NSString *name;
@end
#ViewController.m文件中导入两个类的头文件
#import "ViewController.h"
#import "Person.h"
#import "Teacher.h"
@interface ViewController ()
/**<#注释#>*/
@property (nonatomic , strong) Person *per;
/**<#注释#>*/
@property (nonatomic , strong) Teacher *tea;
@end
- (void)viewDidLoad {
[super viewDidLoad];
self.per = [[Person alloc]init];
self.tea = [[Teacher alloc]init];
//1.注册监听
//我对tea对象注册了一个观察者self.per ,观察tea的name属性
/**
options:
NSKeyValueObservingOptionNew :change那边为新值
NSKeyValueObservingOptionOld :旧值
NSKeyValueObservingOptionInitial:原始值
NSKeyValueObservingOptionPrior :
**/
[self.tea addObserver:self.per forKeyPath:@“name" options:NSKeyValueObservingOptionNew context:nil];
}
#点击屏幕给name赋值
-(void)touchesBegan:(NSSet<UITouch *> *)touches withEvent:(UIEvent *)event{
self.tea.name = @"Tmac";
//在person的.m中去实现一个监听方法
}
其次在Person.m中实现监听方法
//3.监听到object对象的keyPath属性的变化值为change
-(void)observeValueForKeyPath:(NSString *)keyPath ofObject:(id)object change:(NSDictionary<NSKeyValueChangeKey,id> *)change context:(void *)context{
//change这个字典里面固定有两个key:kind 和 一个枚举(这个枚举就是那边注册监听时,options时选的枚举)
NSLog(@"change: %@",change[@"new"]);
}
这样一个简单地KVO监听就实现了,那么这个内部底层到底是怎么做的呢
#其实这是在注册监听的时候,系统利用runtime会动态的生成一个类,类名为:NSKVONotifying_Teacher的子类继承自Teacher类,并在这个类的.m文件中重写父类的name属性的set方法
-(void)setAge:(int)age{
[super setAge:age];
//在子类中会调用这两个方法--还会调用观察者中的observeValueForKeyPath:方法
//willChangeValueForKey:是属性的旧值,did是新值
[self willChangeValueForKey:@“name"];
[self didChangeValueForKey:@“name"];
}
具体细节还需要深入去研究runtime这个”高大上”的东东,面试问到KVO的简单底层原理就可以这样回答