美文网首页
iOS runtime自定义实现KVO

iOS runtime自定义实现KVO

作者: 魔力双鱼 | 来源:发表于2021-05-22 15:57 被阅读0次

1、了解KVO

@interface ViewController ()
@property (nonatomic,strong)Person *p1;
@property (nonatomic,strong)Person *p2;
@end

@implementation ViewController

- (void)observeValueForKeyPath:(NSString *)keyPath ofObject:(id)object change:(NSDictionary<NSKeyValueChangeKey,id> *)change context:(void *)context{
    NSLog(@"变了---key:%@ ---change:%@",keyPath,change);
}


- (void)viewDidLoad {
    [super viewDidLoad];
    // Do any additional setup after loading the view.
    
    _p1 = [[Person alloc]init];
    _p2 = [[Person alloc]init];
    _p1.personName = @"lily";
    _p2.personName = @"lucy";
    NSLog(@"监听之前---p1:%p ---p2:%p",[_p1 methodForSelector:@selector(setPersonName:)],[_p2 methodForSelector:@selector(setPersonName:)]);
    
    [_p1 addObserver:self forKeyPath:@"personName" options:NSKeyValueObservingOptionNew context:nil];
    _p1.personName = @"macBookPro";
    
    /*
     (lldb) po _p1
     <Person: 0x6000033b43f0>

     (lldb) po object_getClassName(_p1)
     "NSKVONotifying_Person"
     */
    NSLog(@"监听之后---p1:%p ---p2:%p",[_p1 methodForSelector:@selector(setPersonName:)],[_p2 methodForSelector:@selector(setPersonName:)]);
}

打印结果:

2021-05-22 15:54:52.280855+0800 hahahah[20817:1289137] 监听之前---p1:0x104136c90 ---p2:0x104136c90
2021-05-22 15:54:52.281967+0800 hahahah[20817:1289137] 变了---key:personName ---change:{
    kind = 1;
    new = macBookPro;
}
2021-05-22 15:54:52.282090+0800 hahahah[20817:1289137] 监听之后---p1:0x7fff207bbb57 ---p2:0x104136c90

2、自定义实现KVO

.h

#import <Foundation/Foundation.h>

NS_ASSUME_NONNULL_BEGIN

@interface Person : NSObject
@property (nonatomic,copy)NSString *personName;
@end

NS_ASSUME_NONNULL_END

.m

#import "Person.h"
#import <objc/message.h>

@implementation Person

void setterMethed(id self,SEL _cmd,NSString *key){
    //1、重写父类方法
    struct objc_super superClass = {
        self,
        class_getSuperclass([self class])
    };
    objc_msgSendSuper(&superClass,_cmd,key);
    //2、通知观察者,执行回调方法
    id observer = objc_getAssociatedObject(self, (__bridge const void *)@"myObjc");
    //key--
    NSString *methedName = NSStringFromSelector(_cmd);
    NSString *myKey = getMethedName(methedName);
    objc_msgSend(observer, @selector(observeValueForKeyPath:ofObject: change:context:),myKey,self,@{myKey:key},nil);
}

NSString * getMethedName(NSString *setter){
    NSRange range = NSMakeRange(3, setter.length-4);
    NSString *key = [setter substringWithRange:range];
    NSString *letter = [[key substringToIndex:1]lowercaseString];
    key = [key stringByReplacingCharactersInRange:(NSMakeRange(0, 1)) withString:letter];
    return key;
}

- (void)hh_addObserver:(NSObject *)observer forKeyPath:(NSString *)keyPath options:(NSKeyValueObservingOptions)options context:(void *)context{
    
    NSString *oldClassStr = NSStringFromClass([self class]);
    NSString *newClassStr = [NSString stringWithFormat:@"hhKVO_%@",oldClassStr];
    
    //1、初始化类对
    Class newClass = objc_allocateClassPair([self class], newClassStr.UTF8String, 0);
    //2、注册类对
    objc_registerClassPair(newClass);
    //3、修改isa
    object_setClass(self, newClass);
    //4、重写set方法
    NSString *newSetMethed = [NSString stringWithFormat:@"set%@",keyPath.capitalizedString];
    SEL sel = NSSelectorFromString(newSetMethed);
    class_addMethod(newClass, sel, (IMP)setterMethed, "v@:@");
    
    objc_setAssociatedObject(self, (__bridge const void *)@"myObjc", observer, OBJC_ASSOCIATION_ASSIGN);
}

@end

相关文章

  • iOS 自定义KVO

    利用Runtime 实现简单的自定义kvo 代码githubgithub.com/zswj/custom-KVO ...

  • iOS 自定义KVO

    自己实现kvo之前,需要知道iOS系统对kvo的实现。 系统实现kvo的原理 这依赖了OC强大的runtime特性...

  • KVO 的本质?

    iOS 用什么方式实现对一个对象的 KVO ? (KVO 的本质是什么?) 首先利用 Runtime API 动态...

  • KVO应用、原理及自实现

    一.KVO简介 KVO 是ios里,观察者设计模式的一种应用实现,依赖runtime,基于KVC,KVO提供了一种...

  • iOS面试题03-Foundation(★★★)

    《2018 iOS面试题系列》 一、KVO内部实现原理 KVO是基于runtime机制实现的 当某个类的对象第一次...

  • iOS Runtime学习笔记 (二) - 实战应用

    iOS runtime实战应用 iOS runtime 进行添加属性,并支持KVO监听 iOS 中category...

  • 面试题(2)

    runtime/KVO等面试题 1.KVO内部实现原则 回答:1>KVO是基于runtime机制实现的 2>当某个...

  • iOS runtime自定义实现KVO

    1、了解KVO 打印结果: 2、自定义实现KVO .h .m

  • iOS-底层原理-自定义KVO

    1.自定义KVO 1.上一篇博客了解了iOS 系统KVO的底层实现原理,那么这里进行自定义KVO,更好的理解原理和...

  • ios-Runtime(运行时)

    利用runtime来实现归档解档 方法交换 俗称 OC的方法欺骗 KVO的实现原理 用runtime来实现KVO...

网友评论

      本文标题:iOS runtime自定义实现KVO

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