KVO简单原理解释

首先我们创建两个类一个Person类和一个Teacher类,在Teacher类中有个name属性
#import 
@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 *)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 *)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的简单底层原理就可以这样回答

你可能感兴趣的:(KVO简单原理解释)