自定义KVO

首先了解一下系统的KVO实现原理:其实就是动态的创建了一个被观察者的子类,然后动态修改它的isa指针指向它的子类,在子类里重写属性的set方法,最后在set方法里监听属性变化,并发出通知。

1、验证系统原理:

image.png

image.png

打个断点,发现实例s的isa是指向Student的,然后单步执行一下,
image.png

image.png

这时候发现实例s的isa指针变成了NSKVONotifying_Student,由此可见,当我们添加观察者的时候,系统动态的创建了一个子类NSKVONotifying_Student,并把s的类型修改成了它。

2、接下来我们自己用RunTime来仿照系统KVO原理来自己写一个KVO

首先创建一个NSObject的分类NSObject+KVO,然后自己实现一个监听方法。

#import "NSObject+KVO.h"
#import <objc/message.h>

static const char* SJKVOAssiociateKey = "SJKVOAssiociateKey";

@implementation NSObject (KVO)
-(void)SJ_addObserver:(NSObject *)observer forKeyPath:(NSString *)keyPath options:(NSKeyValueObservingOptions)options context:(void *)context {
    
    Class newClass = [self createClass:keyPath];

    object_setClass(self, newClass);
    
    // 4.将观察者与对象绑定
    objc_setAssociatedObject(self, SJKVOAssiociateKey, observer, OBJC_ASSOCIATION_ASSIGN);
    
}
- (Class) createClass:(NSString*) keyPath {
    
    // 1. 拼接子类名 / SJKVO_Student
    NSString* oldName = NSStringFromClass([self class]);
    NSString* newName = [NSString stringWithFormat:@"SJKVO_%@", oldName];
    
    // 2. 创建并注册类
    Class newClass = NSClassFromString(newName);
    if (!newClass) {
        
        // 创建并注册类
        newClass = objc_allocateClassPair([self class], newName.UTF8String, 0);
        objc_registerClassPair(newClass);
        
        // 动态添加方法
        // class
        Method classMethod = class_getInstanceMethod([self class], @selector(class));
        const char* classTypes = method_getTypeEncoding(classMethod);
        class_addMethod(newClass, @selector(class), (IMP)SJ_class, classTypes);

    }
    
    // setter
    NSString* setterMethodName = getSetter(keyPath);
    SEL setterSEL = NSSelectorFromString(setterMethodName);
    Method setterMethod = class_getInstanceMethod([self class], setterSEL);
    const char* setterTypes = method_getTypeEncoding(setterMethod);
    
    class_addMethod(newClass, setterSEL, (IMP)SJ_setKey, setterTypes);

    return newClass;
}

Class SJ_class(id self, SEL _cmd) {
    return class_getSuperclass(object_getClass(self));
}

void SJ_setKey(id self, SEL _cmd, id newValue) {
    struct objc_super oldSuper = {self,class_getSuperclass([self class])};
    // 修改属性值
    objc_msgSendSuper(&oldSuper, _cmd, newValue);
    // 拿出观察者
    id observer = objc_getAssociatedObject(self, SJKVOAssiociateKey);
    NSLog(@"---%@",newValue);
    
    // 调用observer
    NSString *methodName = NSStringFromSelector(_cmd);
    NSString *key = getValueKey(methodName);
    
    objc_msgSend(observer, sel_registerName("observeValueForKeyPath:ofObject:change:context:"),key,self,@{key:newValue},nil);
    
//    [observer observeValueForKeyPath:key ofObject:self change:@{key:newValue} context:nil];
}
// key -> setter
static NSString  * getSetter(NSString *keyPath){
    
    if (keyPath.length <= 0) { return nil; }
    
    NSString *firstString = [[keyPath substringToIndex:1] uppercaseString];
    NSString *leaveString = [keyPath substringFromIndex:1];
    
    return [NSString stringWithFormat:@"set%@%@:",firstString,leaveString];
}

// cmd -> key
NSString* getKey(NSString * cmd) {
    if (cmd.length <= 0 || ![cmd hasPrefix:@"set"] || ![cmd hasSuffix:@":"]) { return nil;}
    
    NSRange range = NSMakeRange(3, cmd.length-4);
    NSString *getter = [cmd substringWithRange:range];
    NSString *firstString = [[getter substringToIndex:1] lowercaseString];
    getter = [getter stringByReplacingCharactersInRange:NSMakeRange(0, 1) withString:firstString];
    
    return getter;
    
}

@end

然后在用我们自己的方法来添加一下观察者,看是否能观察到

- (void)viewDidLoad {
    [super viewDidLoad];
    
    Student *s = [[Student alloc] init];
    
//    [s addObserver:self forKeyPath:@"age" options:NSKeyValueObservingOptionNew context:nil];
    
    [s SJ_addObserver:self forKeyPath:@"age" options:NSKeyValueObservingOptionNew context:nil];
    
    self.s = s;
}

-(void)observeValueForKeyPath:(NSString *)keyPath ofObject:(id)object change:(NSDictionary<NSKeyValueChangeKey,id> *)change context:(void *)context {
    
    NSLog(@"+++%@",self.s.age);
}

-(void)touchesBegan:(NSSet<UITouch *> *)touches withEvent:(UIEvent *)event {
    static NSInteger i = 0;
    i++;
    self.s.age = [NSString stringWithFormat:@"%ld",i];
}
image.png

通过打印可以看出跟系统的效果一样。

基本都有注释,我就不详解了,里边一些关于runtime的动态函数和消息转发函数,在我之前关于runtime的文章里都有详解,不了解可以参照前几篇文章做基础。

3、用block回调

用系统的回调会一个问题,就是如果观察的属性多了,在回调方法里需要先判断是哪个对象的哪个属性,比较麻烦,但是用block回调的话,就省去了这些麻烦,并且代码逻辑更清晰,更紧凑。接下来代码多了,我们顺便整理封装一下。

首先定义一个block和一个可以有block的构造方法:

typedef void(^ValueChangeBlock)(id observer, NSString* keyPath, id oldValue, id newValue);

@interface NSObject (KVO)
// 系统回调
- (void)SJ_addObserver:(NSObject *)observer forKeyPath:(NSString *)keyPath options:(NSKeyValueObservingOptions)options context:(nullable void *)context;

// block回调
- (void)SJ_addObserver:(NSObject *)observer forKeyPath:(NSString *)keyPath options:(NSKeyValueObservingOptions)options context:(void *)context ValueChangeBlock:(ValueChangeBlock)valueChangeBlock;

@end

然后在.m文件内部创建一个类,来存储要监听的信息:

static const char* SJKVOAssiociateKey = "SJKVOAssiociateKey";

@interface SJInfo : NSObject

@property (nonatomic, weak) NSObject* observer;
@property (nonatomic, strong) NSString* keyPath;
@property (nonatomic, copy) ValueChangeBlock valueChangeBlock;

@end

@implementation SJInfo

- (instancetype) initWithObserver:(NSObject*)observer forKeyPath:(NSString*) keyPath valueChangeBlock:(ValueChangeBlock) block {
    if (self == [super init]) {
        _observer = observer;
        _keyPath = keyPath;
        _valueChangeBlock = block;
    }
    return self;
}

@end

然后在创建是就把参数和block保存到数组里,以便下边拿出来调用:

-(void)SJ_addObserver:(NSObject *)observer forKeyPath:(NSString *)keyPath options:(NSKeyValueObservingOptions)options context:(void *)context ValueChangeBlock:(ValueChangeBlock)valueChangeBlock {

    Class newClass = [self createClass:keyPath];

    object_setClass(self, newClass);

    // 信息保存
    SJInfo* info = [[SJInfo alloc] initWithObserver:observer forKeyPath:keyPath valueChangeBlock:valueChangeBlock];
    NSMutableArray* array = objc_getAssociatedObject(self, SJKVOAssiociateKey);
    if (!array) {
        array = [NSMutableArray array];
        objc_setAssociatedObject(self, SJKVOAssiociateKey, array, OBJC_ASSOCIATION_RETAIN_NONATOMIC);
    }
    [array addObject:info];

}

void SJ_setKey(id self, SEL _cmd, id newValue) {
    
    struct objc_super oldSuper = {self,class_getSuperclass(object_getClass(self))};
    
    // 获取key
    NSString *key = getKey(NSStringFromSelector(_cmd));
    
    // 获取旧值
    id oldValue = objc_msgSendSuper(&oldSuper, NSSelectorFromString(key));
    
    // 修改属性值
    objc_msgSendSuper(&oldSuper, _cmd, newValue);
    
    NSMutableArray* array = objc_getAssociatedObject(self, SJKVOAssiociateKey);
    if (array) {
        for (SJInfo* info in array) {
            if ([info.keyPath isEqualToString:key]) {
                info.valueChangeBlock(info.observer, key, oldValue, newValue);
                return;
            }
        }
    }
}

在setKey方法里拿出数组中的信息,找到对应的key,然后block回调就可以了,这样就可以同时监听多个属性了。

销毁观察者

其实销毁观察者就是把isa指针指从子类回到原来就可以了,我们把他放在dealloc方法里来做比较合适,有两种方法:1.利用hook在创建子类的方法里做方法交换,把dealloc的方法实现指向自己的方法里,然后做isa指回;2.也是在创建子类的时候同时动态添加自己的dealloc方法来做,我们这里就用第二种实现一下:

放在子类的创建方法里,保证只创建一次

        // 添加SJ_Dealloc,销毁观察者
        SEL deallocSEL = NSSelectorFromString(@"dealloc");
        Method deallocMethod = class_getInstanceMethod([self class], deallocSEL);
        const char* deallocTypes = method_getTypeEncoding(deallocMethod);
        class_addMethod(newClass, deallocSEL, (IMP)SJ_Dealloc, deallocTypes);
void SJ_Dealloc(id self, SEL _cmd) {
    // 父类
    Class superClass = [self class];//class_getSuperclass(object_getClass(self));
    
    object_setClass(self, superClass);

}

最后在外边调用一下,非常方便

    [s SJ_addObserver:self forKeyPath:@"name" options:NSKeyValueObservingOptionNew context:nil ValueChangeBlock:^(id observer, NSString *keyPath, id oldValue, id newValue) {
        NSLog(@"oldValue ---- %@, newValue ---- %@", oldValue, newValue);
    }];
    
    [s SJ_addObserver:self forKeyPath:@"age" options:NSKeyValueObservingOptionNew context:nil ValueChangeBlock:^(id observer, NSString *keyPath, id oldValue, id newValue) {
        NSLog(@"oldValue ++++ %@, newValue ++++ %@", oldValue, newValue);
    }];

©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 214,172评论 6 493
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 91,346评论 3 389
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 159,788评论 0 349
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 57,299评论 1 288
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 66,409评论 6 386
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 50,467评论 1 292
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 39,476评论 3 412
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 38,262评论 0 269
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 44,699评论 1 307
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 36,994评论 2 328
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 39,167评论 1 343
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 34,827评论 4 337
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 40,499评论 3 322
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 31,149评论 0 21
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 32,387评论 1 267
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 47,028评论 2 365
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 44,055评论 2 352

推荐阅读更多精彩内容

  • 1.ios高性能编程 (1).内层 最小的内层平均值和峰值(2).耗电量 高效的算法和数据结构(3).初始化时...
    欧辰_OSR阅读 29,353评论 8 265
  • Swift1> Swift和OC的区别1.1> Swift没有地址/指针的概念1.2> 泛型1.3> 类型严谨 对...
    cosWriter阅读 11,094评论 1 32
  • 自己实现kvo之前,需要知道iOS系统对kvo的实现。 系统实现kvo的原理 这依赖了OC强大的runtime特性...
    mws100阅读 2,778评论 6 3
  • 你好热热热
    古筝风阅读 50评论 0 0
  • 有关鲫鱼,记忆中,小的时候经常吃,每每做到鱼汤,爸妈总会先舀出一碗鱼汤让我先喝下去,依旧到现在也是如此。可以说,我...
    狮子猫2017阅读 255评论 0 1