关于iOS Runtime让你了解更多

关于Runtime

Runtime根据字面理解就是运行时,当我们的代码运行的时候所体现的东西,举一个比较简单一点的例子,这里有一个Person类,然后创建一个Student类继承Person,这里我们知道可以用Person类的对象来接收Student类创建的对象,从代码看来这个对象时一个Person对象,但是在代码运行的时候,这个对象体现出来的却是一个Student对象。

Runtime API

获取对象的类
Class object_getClass(id obj) 
设置对象的类
Class object_setClass(id obj, Class cls)
获取对象的类名
const char *object_getClassName(id obj)
获取实例变量的值
id object_getIvar(id obj, Ivar ivar)
设置实例变量的值 这个方法比下面这个设置实例变量要快
void object_setIvar(id obj, Ivar ivar, id value)
设置实例变量的值
Ivar object_setInstanceVariable(id obj, const char *name, void *value)
获取实例变量变量和值
Ivar object_getInstanceVariable(id obj, const char *name, void **outValue)
根据名称获取类
Class objc_getClass(const char *name)
获取对应类和实例变量名的Ivar指针
Ivar class_getInstanceVariable(Class cls, const char *name)
获取类对应的实例变量的Ivar指针数组
Ivar *class_copyIvarList(Class cls, unsigned int *outCount) 
根据类名和SEL变量获取实例方法对象
Method class_getInstanceMethod(Class cls, SEL name)
根据类名和SEL变量获取类方法对象
Method class_getClassMethod(Class cls, SEL name)
根据类和方法名获取IMP指针 这个方法比IMP method_getImplementation(Method m) 可能要快一点
IMP类型是(void(*)(id instance, SEL _cmd, id parameter1,...))
IMP class_getMethodImplementation(Class cls, SEL name)
类实例能否响应这个SEL
BOOL class_respondsToSelector(Class cls, SEL sel)
根据name获取属性
objc_property_t class_getProperty(Class cls, const char *name)
获取属性列表
objc_property_t *class_copyPropertyList(Class cls, unsigned int *outCount)
给指定类添加新方法
class_addMethod(Class cls, SEL name, IMP imp, 
                                 const char *types) 
取代一个方法的实现
IMP class_replaceMethod(Class cls, SEL name, IMP imp, 
                                    const char *types)
添加一个实例变量
这里举一个例子
class_addIvar(newClass, "name", sizeof(NSString *), log2(sizeof(NSString *)), @encode(NSString *));
BOOL class_addIvar(Class cls, const char *name, size_t size, 
                               uint8_t alignment, const char *types)
添加一个属性
attributes这个参数的顺序要保持T(Type)在第一位,V(Ivar)在最后一位
BOOL class_addProperty(Class cls, const char *name, const objc_property_attribute_t *attributes, unsigned int attributeCount)
替换属性
void class_replaceProperty(Class cls, const char *name, const objc_property_attribute_t *attributes, unsigned int attributeCount)
创建一个新类
Class objc_allocateClassPair(Class superclass, const char *name, 
                                         size_t extraBytes)
注册这个新类 给创建的新类添加属性 方法 协议 成员变量等都需要在注册之前完成
void objc_registerClassPair(Class cls)
Method转SEL变量
SEL method_getName(Method m)
Method转IMP指针
IMP method_getImplementation(Method m)
设置一个Method的IMP指针 返回之前的IMP指针
IMP method_setImplementation(Method m, IMP imp)
交换两个Method
void method_exchangeImplementations(Method m1, Method m2)
获取变量名
const char *ivar_getName(Ivar v) 
获取变量的类型编码
const char *ivar_getTypeEncoding(Ivar v)
获取属性名
const char *property_getName(objc_property_t property)
获取属性的特性
const char *property_getAttributes(objc_property_t property)
获取属性特性数组
objc_property_attribute_t *property_copyAttributeList(objc_property_t property, unsigned int *outCount)
根据属性的特性name获取value
char *property_copyAttributeValue(objc_property_t property, const char *attributeName)
SEL转c字符串
const char *sel_getName(SEL sel)
根据str返回一个SEL变量
SEL sel_getUid(const char *str)
根据str注册一个SEL变量
SEL sel_registerName(const char *str)
判断两个SEL变量是否相等
BOOL sel_isEqual(SEL lhs, SEL rhs)
block转IMP
block举例 由于是block所以没有SEL参数
id block = ^(id instance, id parameter1, id parameter2) {
            
        };
IMP imp_implementationWithBlock(id block)
IMP转block
id imp_getBlock(IMP anImp)
给一个对象关联一个指定的key和关联方式
void objc_setAssociatedObject(id object, const void *key, id value, objc_AssociationPolicy policy)
获取关联值
id objc_getAssociatedObject(id object, const void *key)
移除关联值
void objc_removeAssociatedObjects(id object)

IMP SEL Method是可以相互转化的,下面上一张图,做的不怎么好的,凑合着看吧


IMP&SEL&Method

Type Encoding

Type Encoding

关于属性的特性我这里放一个链接
objc_property_attribute_t

实际应用

给Category添加属性

我就随便给一个Category添加一个莫名其妙的属性

static const void *countKey = &countKey;

@implementation NSObject (Count)

- (void)setCount:(NSInteger)count {
    objc_setAssociatedObject(self, countKey, @(count), OBJC_ASSOCIATION_ASSIGN);
}

- (NSInteger)count {
    return [objc_getAssociatedObject(self, countKey) integerValue];
}

@end
拦截系统的方法(swizzling)

我这里根据IMP SEL Method的关系,我想到三种实现的方法

static IMP _setColor;

static const void *pageKey = &pageKey;
static const void *nameKey = &nameKey;

@implementation UIView (Border)

+ (void)load {
/***********************************交换方法************************/
    //方法1
//    Method method = class_getInstanceMethod(self, @selector(setBackgroundColor:));
//    _setColor = method_setImplementation(method, (IMP)colorOfBackground);
    //方法2
    _setColor = class_replaceMethod(self, @selector(setBackgroundColor:), (IMP)colorOfBackground, "v@:@");
    //方法3
//    Method method1 = class_getInstanceMethod(self, @selector(setBackgroundColor:));
//    Method method2 = class_getInstanceMethod(self, @selector(colorForBackgroundColor:));
//    method_exchangeImplementations(method1, method2);
/***********************************交换方法************************/
}

- (void)colorForBackgroundColor:(UIColor *)color {
//    这里不会循环,因为已经交换方法,调用本身相当于调用系统setBackgroundColor:方法
    [self colorForBackgroundColor:color];
    NSLog(@"设置颜色成功, 并替换了方法 ---colorForBackgroundColor");
}

void colorOfBackground(UIView *view, SEL sel, UIColor *color) {
    ((void (*)(UIView *, SEL, UIColor *))_setColor)(view, sel, color);
    NSLog(@"设置颜色成功, 并替换了方法 ---colorOfBackground");
}
快速序列化,实现归档和反归档
- (instancetype)initWithCoder:(NSCoder *)aDecoder {
    if (self = [super init]) {
        unsigned int outCount;
        Ivar * ivars = class_copyIvarList([self class], &outCount);
        for (int i = 0; i < outCount; i ++) {
            Ivar ivar = ivars[i];
            NSString * key = [NSString stringWithUTF8String:ivar_getName(ivar)];
            [self setValue:[aDecoder decodeObjectForKey:key] forKey:key];
        }
    }
    return self;
}

- (void)encodeWithCoder:(NSCoder *)aCoder {
    unsigned int outCount;
    Ivar * ivars = class_copyIvarList([self class], &outCount);
    for (int i = 0; i < outCount; i ++) {
        Ivar ivar = ivars[i];
        NSString * key = [NSString stringWithUTF8String:ivar_getName(ivar)];
        [aCoder encodeObject:[self valueForKey:key] forKey:key];
    }
}
实现自己的KVO

如果要实现自己的KVO就要先了解苹果自带的KVO的实现原理,相信大家都会用KVO,在调用- (void)addObserver:(NSObject *)observer forKeyPath:(NSString *)keyPath options:(NSKeyValueObservingOptions)options context:(nullable void *)context;这个方法的时候,会发现当前这个调用者的isa指针发生变化,这里运用了Runtime的东西。实现自己的KVO重点在于重写setter方法和消息的转发。调用msg_Send()是出现错误,在BuildSetting,搜索msg,将YES改为NO
这里为NSObject添加了一个分类,其实系统的KVO也是NSObject的一个分类实现

//
//  NSObject+FSKVO.m
//  Runtime
//
//  Created by vcyber on 17/8/23.
//  Copyright © 2017年 vcyber. All rights reserved.
//

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

static const void *observerKey = &observerKey;
static const void *keyPathKey = &keyPathKey;
static const void *optionsKey = &optionsKey;
static const void *setterKey = &setterKey;
static const void *oldValueKey = &oldValueKey;

@implementation NSObject (FSKVO)

- (void)fs_addObserver:(NSObject *)observer forKeyPath:(NSString *)keyPath options:(NSKeyValueObservingOptions)options context:(void *)context {
    //1.动态生成一个类
    NSString *oldClassName = NSStringFromClass([self class]);
    NSString *newClassName = [@"FS_" stringByAppendingString:oldClassName];
    Class newClass = objc_allocateClassPair([self class], newClassName.UTF8String, 0);
    //2.为新生成的类添加setter方法
    NSString *setter = [@"set" stringByAppendingString:[[keyPath capitalizedString] stringByAppendingString:@":"]];
    class_addMethod(newClass, NSSelectorFromString(setter), (IMP)setKeyPath, "v@:@");
    objc_registerClassPair(newClass);
    //修改被观察者的isa指针!!让它指向自定义的类!!
    object_setClass(self, newClass);
    //3.获取旧值
    id oldValue = [self valueForKeyPath:keyPath];
    //3.保存observer keyPath options setter 旧值 用于消息发送
    objc_setAssociatedObject(self, observerKey, observer, OBJC_ASSOCIATION_RETAIN_NONATOMIC);
    objc_setAssociatedObject(self, keyPathKey, keyPath, OBJC_ASSOCIATION_COPY);
    objc_setAssociatedObject(self, optionsKey, @(options), OBJC_ASSOCIATION_ASSIGN);
    objc_setAssociatedObject(self, setterKey, setter, OBJC_ASSOCIATION_COPY);
    objc_setAssociatedObject(self, oldValueKey, oldValue, OBJC_ASSOCIATION_RETAIN_NONATOMIC);
}


void setKeyPath(id self, SEL _cmd, id newValue) {
    id class = [self class];
    //改变当前对象指向父类!!
    object_setClass(self, class_getSuperclass(class));
    //调用父类的setter方法
    
    NSString *setter = objc_getAssociatedObject(self, setterKey);
    objc_msgSend(self, NSSelectorFromString(setter), newValue);
    //取出观察者 转发消息
    id observer = objc_getAssociatedObject(self, observerKey);
    NSString *keyPath = objc_getAssociatedObject(self, keyPathKey);
    NSUInteger options = [objc_getAssociatedObject(self, optionsKey) unsignedIntegerValue];
    NSMutableDictionary<NSKeyValueChangeKey, id> *change = [NSMutableDictionary dictionary];
    if (options & NSKeyValueObservingOptionNew) {
        change[NSKeyValueChangeNewKey] = newValue;
    }
    if (options & NSKeyValueObservingOptionOld) {
        change[NSKeyValueChangeOldKey] = objc_getAssociatedObject(self, oldValueKey);
    }
    objc_msgSend(observer, @selector(observeValueForKeyPath:ofObject:change:context:), keyPath, self, change, NULL);
    object_setClass(self, class);
    objc_setAssociatedObject(self, oldValueKey, newValue, OBJC_ASSOCIATION_RETAIN_NONATOMIC);
}


@end

用法和系统的用法一样

    MyClass1 *class1 = [[MyClass1 alloc] init];
    _cls = class1;
    class1.string = @"test";
    [class1 fs_addObserver:self forKeyPath:@"string" options:NSKeyValueObservingOptionNew | NSKeyValueObservingOptionOld context:NULL];

- (void)observeValueForKeyPath:(NSString *)keyPath ofObject:(id)object change:(NSDictionary<NSKeyValueChangeKey,id> *)change context:(void *)context {
    NSLog(@"%@-%@", keyPath, change);
}
效果
JSON转模型

这里我就不做简单的演示了(因为难的我也不会),大家可以看看别人的三方库,里面用到了runtime的东西。

总结

大家还有什么好的runtime使用情况,可以留言或者私信给我,我就添加在文章上面,供给大家看。欢迎吐槽!!!

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

推荐阅读更多精彩内容