关于iOS Runtime让你了解更多

作者: Fly_Sunshine_J | 来源:发表于2017-08-23 18:18 被阅读84次

关于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使用情况,可以留言或者私信给我,我就添加在文章上面,供给大家看。欢迎吐槽!!!

相关文章

网友评论

    本文标题:关于iOS Runtime让你了解更多

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