美文网首页
KVO使用及实现原理

KVO使用及实现原理

作者: Corbin___ | 来源:发表于2019-01-12 14:14 被阅读0次

KVO使用及实现原理

KVO使用

  • 对属性进行监听
  • 对属性的属性进行监听
  • 容器监听
  • 触发(手动触发,kvc赋值)

添加监听


    // 1.kvo对属性的监听
    [_person addObserver:self forKeyPath:NSStringFromSelector(@selector(name)) options:NSKeyValueObservingOptionNew context:nil];
    // 2.kvo属性关联,这个我们是监听dog的变化,那如果是dog属性的变化,如果不做处理,是监听不到的
    [_person addObserver:self forKeyPath:NSStringFromSelector(@selector(dog)) options:NSKeyValueObservingOptionNew context:nil];
    // 3.容器监听
    [_person addObserver:self forKeyPath:NSStringFromSelector(@selector(mArr)) options:NSKeyValueObservingOptionNew context:nil];

触发代码

    // 触发基本知识点:KVO得通过set方法才可以触发
    
     NSString *name = NSStringFromSelector(@selector(name));
    
    // 手动触发 这两个方法必须是成对的,然后回调observeValueForKeyPath方法,至于为什么成对呢,猜想应该是苹果做了处理,点进去官方的注释也有说明这两个方法必须成对存在
    // 如果想要手动触发需要在被监听类中实现+ (BOOL)automaticallyNotifiesObserversForKey:(NSString *)key返回NO
    [_person willChangeValueForKey:name];
    [_person didChangeValueForKey:name];
    
    // kvc
    [_person setValue:@"123" forKey:name];
    
    // 容器
    NSMutableArray *arr1 = [_person mutableArrayValueForKey:NSStringFromSelector(@selector(mArr))];
    [arr1 addObject:@"123"];
    
    NSMutableArray *arr2 = [_person mutableArrayValueForKey:NSStringFromSelector(@selector(mArr))];
    [arr2 replaceObjectAtIndex:0 withObject:@"1234"];
    
    NSMutableArray *arr3 = [_person mutableArrayValueForKey:NSStringFromSelector(@selector(mArr))];
    [arr3 removeAllObjects];
    

    
    // 属性关联
    // 属性关联需要在被监听类中实现+ (NSSet<NSString *> *)keyPathsForValuesAffectingValueForKey:(NSString *)key
    _person.dog.name = @"myDog";

被监听类中实现的代码

// 这个方法不重写,就是默认返回YES,如果重写了返回NO,那么就是需要手动触发了,当然这个可以根据参数key来判断,区分监听的key是手动还是自动 
+ (BOOL)automaticallyNotifiesObserversForKey:(NSString *)key
{
    return YES;
}

// 这个方法用于监听属性的属性的(属性关联)
+ (NSSet<NSString *> *)keyPathsForValuesAffectingValueForKey:(NSString *)key
{
    if ([key isEqualToString:@"dog"]) {
        return [[NSSet alloc] initWithObjects:@"_dog.name", nil];
    } else {
        return [super keyPathsForValuesAffectingValueForKey:key];
    }
}

移除监听

懒得写

KVO原理

1.在运行时的时候创建被监听类的子类

    /** 1.动态生成一个类 */
    /** 1.1 动态生成一个类名 */
    NSString *oldClassName = NSStringFromClass(self.class);
    NSString *newClassName = [@"KVO_" stringByAppendingString:oldClassName];
    /** 定义一个类,继承传进来的类 */
    Class MyClass = objc_allocateClassPair(self.class, newClassName.UTF8String, 0);

2.在子类中重写父类属性的set方法(所以KVO只能监听属性)

    /** 添加set方法 */
    class_addMethod(MyClass, @selector(setName:), (IMP)setName, "V@:@");

3.注册这个子类

    /** 注册该类 */
    objc_registerClassPair(MyClass);

4.修改当前被监听对象的isa指针指向子类

    /** 修改isa指针 */
    object_setClass(self, MyClass);

5.实现set函数


/** set方法 */
void setName(id self,SEL _cmd,NSString * newName){
    /** 保存当前类型 */
    Class class = [self class];
    /** 调用父类方法 */
    object_setClass(self, class_getSuperclass(class));
    objc_msgSend(self, @selector(setName:),newName);
    /** 通知观察者 */
    // 这里有个属性绑定,所以在添加观察者的时候,就是将这个观察者持有(就是使用属性绑定来持有)
    id observer = objc_getAssociatedObject(self, @"objc");
    if (observer) {
        objc_msgSend(observer, @selector(observeValueForKeyPath:ofObject:change:context:),@"name",self,@{@"new:":[self valueForKey:@"name"],@"kind:":@1},nil);
    }
    /** 改回子类 */
     object_setClass(self, class);
     
    // kvo就是在set方法中调用
    willChangeValueForKey:
    didChangeValueForKey:;
    那我们这里需不需要呢,当然不需要,因为我们是自己实现了一套了,调这两句也没有用的
}

思考点

1.为什么kvc可以触发,kvc的原理

2.创建一个子类,如果创建的子类的类名,项目中刚好存在呢

如果项目中存在,那个创建的子类会nil,那么这时候我们可以使用递归创建,为nil就在类名后拼1或者其他字符吧,直到成功为止

3.objc_msgSend(self, @selector(setName:),newName);可以传值,那这个跟performSelector:有什么关系,而且发现直接用objc_msgSend还比performSelector方便

4.为什么oc的每个方法都有两个隐式参数,这两个是哪里来的

oc调用方法是消息机制,表现形式就是objc_msgSend(self, @selector(setName:),newName);那么每个方法在调用的本质都是使用objc_msgSend那这个刚好就需要传两个参数,调用者和方法编号,也就是isa和sel

相关文章

  • KVO使用及实现原理

    KVO使用及实现原理 KVO使用 对属性进行监听 对属性的属性进行监听 容器监听 触发(手动触发,kvc赋值) 添...

  • KVO使用及实现原理

    KVO是什么 KVO:全称Key-Value Observing,键值观察机制,是苹果提供的一套事件通知机制,也是...

  • iOS高级进阶之KVO

    KVO的原理 分析原理 使用 手动调用 自己实现KVO NSObject+KVOBlock.h NSObject+...

  • iOS原理篇(一): KVO实现原理

    KVO实现原理 什么是 KVO KVO 基本使用 KVO 的本质 总结 一 、 什么是KVO KVO(Key-Va...

  • KVO和KVC的使用及原理解析

    一 KVO基本使用 二 KVO本质原理讲解及代码验证 三 KVC基本使用 四 KVC设值原理 五 KVC取值原理 ...

  • iOS探索KVO实现原理,重写KVO

    写响应式编程博客时,提到了KVO,今天我们探索一下KVO的实现原理及如何自己实现KVO功能 首先简单的KVO实现 ...

  • KVO基本使用

    分三部分解释KVO一.KVO基本使用二.KVO原理解析三.自定义实现KVO 一、KVO基本使用 使用KVO,能够非...

  • iOS KVO机制的基本使用及实现原理

    我用#CSDN#《iOS KVO机制的基本使用及实现原理》 https://blog.csdn.net/u0112...

  • iOS-自定义KVO

    KVO原理及使用 我们之前讨论过KVO的原理,知道KVO机制是生成了一个中间类NSKVONotifying,该中间...

  • KVO和KVC的理解

    KVO键值监听的使用 KVO是OC观察者模式的又一实现,使用了isa混写来实现的KVO KVO原理 1.运行时会创...

网友评论

      本文标题:KVO使用及实现原理

      本文链接:https://www.haomeiwen.com/subject/pofcdqtx.html