OC语法(KVO)

KVO的全称是Key-Value Observing,即"键值监听",可以用于监听某个对象属性值得改变。


F、题:iOS用什么方式实现对一个对象的KVO?(KVO的本质是什么)

答:

  1. 当一个对象使用了KVO监听,iOS系统会修改这个对象的isa指针,改为指向一个全新的通过Runtime动态创建的子类(NSKVONotifying_xxx)。
  2. 子类拥有自己的set方法实现,set方法实现内部会顺序调用willChangeValueForKey方法、[super setAxx:]didChangeValueForKey方法。
  3. didChangeValueForKey方法内部又会调用监听器的observeValueForKeyPath:ofObject:change:context:方法
  4. 这几个方法合起来就是Foundation的C方法_NSSetInt(Double)ValueAndNotify方法。

拓展:
//给person对象添加KVO监听
NSKeyValueObservingOptions options = NSKeyValueObservingOptionNew | NSKeyValueObservingOptionOld;
[self.person addObserver:self forKeyPath:@"age" options:options context:@"123"];
[self.person addObserver:self forKeyPath:@"height" options:options context:@"456"];

- (void)observeValueForKeyPath:(NSString *)keyPath ofObject:(id)object change:(NSDictionary *)change context:(void *)context {
    NSLog(@"监听到%@的%@属性值改变了 - %@---content-%@", object, keyPath, change, context);
}

- (void)dealloc {
    [self.person removeObserver:self forKeyPath:@"age"];
    [self.person removeObserver:self forKeyPath:@"height"];
}

2019-01-26 17:22:15.315762+0800 InterViewOC[63236:684490] 监听到的age属性值改变了 - {
    kind = 1;
    new = 20;
    old = 10;
}---content-123
2019-01-26 17:22:15.315972+0800 InterViewOC[63236:684490] 监听到的height属性值改变了 - {
    kind = 1;
    new = 21;
    old = 11;
}---content-456

题:如何手动触发KVO

答:被监听的属性的值被修改时,就会自动触发KVO。如果想要手动触发KVO,则需要我们手动调用willChangeValueForKey和didChaneValueForKey方法即可在不改变属性值的情况下手动触发KVO,并且这两个方法缺一不可。(应该是在didChaneValueForKey方法中判断过前面是否调用过willChangeValueForKey,所以两者缺一不可)

LQPerson *p1 = [[LQPerson alloc] init];
p1.age = 1.0;
   
NSKeyValueObservingOptions options = NSKeyValueObservingOptionNew | NSKeyValueObservingOptionOld;
[p1 addObserver:self forKeyPath:@"age" options:options context:@"789"];
    
[p1 willChangeValueForKey:@"age"];
[p1 didChangeValueForKey:@"age"];
    
[p1 removeObserver:self forKeyPath:@"age"];

2019-01-26 17:22:15.315972+0800 InterViewOC[63236:684490] 监听到的height属性值改变了 - {
    kind = 1;
    new = 1;
    old = 1;
}---content-789

注意:不是每个属性都能监听的,这个取决于代码如何实现的。

题:直接修改成员变量会触发KVO么?

答:不会触发KVO,因为直接修改成员变量并没有调用setXxx方法。
除非使用[self willChangeValueForKey:@"xxx"]; 修改成员变量; [self didChangeValueForKey:@"xxx"]; 才会触发KVO。

_NSSet*ValueAndNotify的内部调用流程:
  1. [self willChangeValueForKey:@"xxx"];   // 调用willChangeValueForKey
    [super setXxx:];  // 原来的setter实现
    [self didChangeValueForKey:@"xxx"];    //  调用didChangeValueForKey
    
  2. didChangeValueForKey内部会调用observerObserverValueForKeyPath:ofObject:change:context

  3. - (void)willChangeValueForKey:(NSString *)key {
      NSLog(@"willChangeValueForKey  ---  begin");
      [super willChangeValueForKey:key];
      NSLog(@"willChangeValueForKey  ---  end");
    }
    
    - (void)didChangeValueForKey:(NSString *)key {
      NSLog(@"didChangeValueForKey  ---  begin");
      [super didChangeValueForKey:key];
      NSLog(@"didChangeValueForKey  ---  end");
    }
    
    
打印类对象中的方法列表:
- (void)printMethodWithClass:(Class)cls {
    unsigned int count;
    // 获得方法数组
    Method *methods = class_copyMethodList(cls, &count);
    // 存储方法名
    NSMutableString *methodNames = [NSMutableString string];
    // 遍历所有方法
    for (int i = 0; i < count; i++) {
        // 获得方法
        Method method = methods[i];
        // 获得方法名
        NSString *mname = NSStringFromSelector(method_getName(method));
        // 拼接方法名
        [methodNames appendFormat:@"%@ ", mname];
    }
    // 释放内存
    free(methods);
}

你可能感兴趣的:(OC语法(KVO))