美文网首页
iOS - 自定义KVO

iOS - 自定义KVO

作者: ForScanf | 来源:发表于2018-12-26 16:14 被阅读4次

    之前我们已经了解过了KVO的底层实现原理,不过呢,在我们开始实现自定义KVO之前再来简单回顾下KVO的实现原理

    1.创建子类
    2.重写一个setter方法(其实是添加一个setter方法)
    3.修改isa指针指向新创建的子类
    4.调用父类的setName方法
    5.将观察者保存到当前对象

    接下来我们开始自定义KVO

    我们先创建一个NSObject分类;然后实现对应的方法

    #import <Foundation/Foundation.h>
    
    NS_ASSUME_NONNULL_BEGIN
    
    @interface NSObject (KVO)
    
    - (void)ZXY_addObserver:(NSObject *)observer forKeyPath:(NSString *)keyPath options:(NSKeyValueObservingOptions)options context:(nullable void *)context;
    
    @end
    
    NS_ASSUME_NONNULL_END
    

    1.创建子类

    在.m文件里#import <objc/message.h>
    创建完子类记得要注册这个类,

    - (void)ZXY_addObserver:(NSObject *)observer forKeyPath:(NSString *)keyPath options:(NSKeyValueObservingOptions)options context:(void *)context{
        
        //1.创建子类
        NSString *oldClassName = NSStringFromClass(self.class);
        NSString *newClassName = [@"ZXYKVO_" stringByAppendingString:oldClassName];
        
        Class myClass = objc_allocateClassPair(self.class, newClassName.UTF8String, 0);
        
        // 注册
        objc_registerClassPair(myClass);    
        
    }
    

    2.重写一个setter方法(其实是添加一个setter方法)

    - (void)ZXY_addObserver:(NSObject *)observer forKeyPath:(NSString *)keyPath options:(NSKeyValueObservingOptions)options context:(void *)context{
        
        //1.创建子类
        NSString *oldClassName = NSStringFromClass(self.class);
        NSString *newClassName = [@"ZXYKVO_" stringByAppendingString:oldClassName];
        
        Class myClass = objc_allocateClassPair(self.class, newClassName.UTF8String, 0);
        
        // 注册
        objc_registerClassPair(myClass);
    
        //2.重新setNanme方法(其实是添加一个setName方法)
        class_addMethod(myClass, @selector(setName:), (IMP)setMethod, "v@:@");
    }
    
    void setMethod(id self,SEL _cmd,NSString *newName){
       
        NSLog(@"来了%@",newName);
    }
    
    

    3.修改isa指针指向新创建的子类

    - (void)ZXY_addObserver:(NSObject *)observer forKeyPath:(NSString *)keyPath options:(NSKeyValueObservingOptions)options context:(void *)context{
        
        //1.创建子类  
        NSString *oldClassName = NSStringFromClass(self.class);
        NSString *newClassName = [@"ZXYKVO_" stringByAppendingString:oldClassName];
        
        Class myClass = objc_allocateClassPair(self.class, newClassName.UTF8String, 0);
        
        // 注册
        objc_registerClassPair(myClass);
    
        //2.重新setNanme方法(其实是添加一个setName方法)
        class_addMethod(myClass, @selector(setName:), (IMP)setMethod, "v@:@");
    
        //3.修改imp地址
        object_setClass(self, myClass);
    }
    

    这个时候我们在控制器中注册这个观察者( 其中person是需要观察属性变化的实体类)

    - (void)viewDidLoad {
        [super viewDidLoad];
    
        person = [[Person alloc]init];
        [person ZXY_addObserver:self forKeyPath:@"name" options:(NSKeyValueObservingOptionNew) context:nil];
    
    }
    //点击屏幕
    - (void)touchesBegan:(NSSet<UITouch *> *)touches withEvent:(UIEvent *)event{
        
        static int a;
        person.name = [NSString stringWithFormat:@"%d",a++];
    }
    

    然后我们跑起来,点击屏幕: 控制台打印


    屏幕快照 2018-12-25 下午4.33.43.png

    4.调用父类的setName方法

    到目前这个阶段,我们已经能获取到修改的值了,接下来我们需要调用父类的set方法,去修改属性值,这样person的name就修改了.

    void setName(id self,SEL _cmd,NSString *newName){
        
        NSLog(@"来了%@",newName);
        
        //5. 调用父类的setName方法
        Class class = [self class];//拿到当前类型
    
        object_setClass(self, class_getSuperclass(class));// 将当前类型设置为父类类型
    
        objc_msgSend(self, @selector(setName:),newName);//给父类的set方法发送消息,传递修改的值
    
        // 改为子类
        object_setClass(self, class);
    }
    
    注意:在这一步一定要将当前的类型设置为父类类型,设置完后也一定要改为子类的类型

    5.将观察者保存到当前对象

    现在我们需要通知外部方法,已经将属性修改了,并将之传递出去,所以我们设置一个观察者

    - (void)ZXY_addObserver:(NSObject *)observer forKeyPath:(NSString *)keyPath options:(NSKeyValueObservingOptions)options context:(void *)context{
        
        //1.创建子类   
        NSString *oldClassName = NSStringFromClass(self.class);
        NSString *newClassName = [@"ZXYKVO_" stringByAppendingString:oldClassName];
        
        Class myClass = objc_allocateClassPair(self.class, newClassName.UTF8String, 0);
        
        // 注册
        objc_registerClassPair(myClass);
    
        //2.重新setNanme方法(其实是添加一个setName方法)
        /**参数
         *class 给哪个类添加方法
         *sel   方法编号
         *imp   方法实现(函数指针)
         *type  返回值类型 ()
         */
        class_addMethod(myClass, @selector(setName:), (IMP)setMethod, "v@:@");
    
        //3.修改imp地址
        object_setClass(self, myClass);
    
        //4.将观察者保存到当前对象
       /**
         id object                     :表示关联者,是一个对象,变量名理所当然也是object
         const void *key               :获取被关联者的索引key
         id value                      :被关联者,这里是一个block
         objc_AssociationPolicy policy : 关联时采用的协议,有assign,retain,copy等协议,一般使用OBJC_ASSOCIATION_RETAIN_NONATOMIC,这里我们使用Weak协议,避免循环引用
         */
        objc_setAssociatedObject(self, "observer", observer, OBJC_ASSOCIATION_ASSIGN);//
    }
    
    在这里就有可能就有人问了,明明set方法只有一个返回类型,为什么要写“v@:@”,这里解释下,V表示的是返回void类型,第二个@表示的是一个id对象,第三个:表示的是方法SEL,最后一个@表示的传入的变量值.
    ⚠️注意:以为OC方法会默认两个参数,一个id对象,一个方法SEL

    然后我们在set方法中获取到这个观察者

    void setName(id self,SEL _cmd,NSString *newName){
    
        NSLog(@"来了%@",newName);
        
        //5. 调用父类的setName方法
        Class class = [self class];//拿到当前类型
    
        object_setClass(self, class_getSuperclass(class));
    
        objc_msgSend(self, @selector(setName:),newName);
    
        // 观察者
        id observer = objc_getAssociatedObject(self, "observer");
    
        // 改为子类
        object_setClass(self, class);
    }
    

    给系统的observeValueForKeyPath方法发送消息

    void setName(id self,SEL _cmd,NSString *newName){
    
        NSLog(@"来了%@",newName);
        
        //5. 调用父类的setName方法
        Class class = [self class];//拿到当前类型
    
        object_setClass(self, class_getSuperclass(class));
    
        objc_msgSend(self, @selector(setName:),newName);
    
        // 观察者
        id observer = objc_getAssociatedObject(self, "observer");
    
        if (observer) {
            objc_msgSend(observer, @selector(observeValueForKeyPath:ofObject:change:context:),@"name",self,@{@"new:":newName,@"kind:":@1},nil);
        }
    
        // 改为子类
        object_setClass(self, class);
    }
    

    其中observeValueForKeyPath方法的参数说明:
    keyPath: 属性名字
    object: 哪个对象
    change: 一个字典,它描述对键路径keyPath中的属性值相对于对象所做的更改
    context: 当注册观察者以接收键值观察通知时提供的值,这里写为nil就行.

    最后

    现在我们在外部控制器中打印修改后的值

    - (void)observeValueForKeyPath:(NSString *)keyPath ofObject:(id)object change:(NSDictionary<NSKeyValueChangeKey,id> *)change context:(void *)context{
        
        NSLog(@"%@",change);
        
    }
    
    屏幕快照 2018-12-26 下午2.23.06.png

    到这里,自定义KVO就简单实现了,但是目前里面的属性变量名都是固定,如果能自定义的变化,那就更完美了.降下来,我们就来慢慢的研究他.

    相关文章

      网友评论

          本文标题:iOS - 自定义KVO

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