oc的运行期环境(runtime)

oc语言的动态特性

oc语言的动态特性来自ObjC Runtime ,其实是一个 runtime 库,基本上用 C 和汇编写的,这个库作用就是加载类的信息,进行方法的分发和转发之类的。具体为对象接受消息(oc是消息型语言)之后,调用何种方法直到运行期才能决定,编译器看到此消息后将其转为标准的C语言函数调用,函数objc_msgSend为消息传递的核心函数; objc_msgSend根据消息选择子的调用适当的方法,在对象所属类中搜寻方法列表,找到之后跳转到实现代码,若找不到会沿着继承体系继续往上找,找到之后再跳转;若始终找不到(无法响应选择子),则启动消息转发机制,包括动态方法解析和消息转发,若经过消息转发流程之后,还是无法处理选择子,则会抛出异常

理解oc中的一些定义和消息发送
//oc中的方法定义
typedef struct objc_selector  *SEL;
SEL aSel = @selector(goHome);

//oc中的类定义
typedef struct objc_class   *Class;
typedef struct objc_object  {
    Class isa; //isa 指向所属的类
} *id;  //id 为指向对象的指针

struct objc_class {
    Class isa; // 指向metaclass
    Class super_class ; // 指向父类
    const char *name ; // 类名
    long version ; // 类的版本信息,初始化默认为0,可以通过runtime函数class_setVersion或者class_getVersion进行修改、读取
    long info; // 一些标识信息,如CLS_CLASS (0x1L) 表示该类为普通 class ,其中包含实例方法和变量;CLS_META (0x2L) 表示该类为 metaclass,其中包含类方法;
    long instance_size ; // 该类的实例变量大小(包括从父类继承下来的实例变量);
    struct objc_ivar_list *ivars; // 用于存储每个成员变量的地址
    struct objc_method_list **methodLists ; // 与 info 的一些标志位有关,如CLS_CLASS (0x1L),则存储实例方法,如CLS_META (0x2L),则存储类方法;
    struct objc_cache *cache; // 指向最近使用的方法的指针,用于提升效率;
    struct objc_protocol_list *protocols; // 存储该类声明遵守的协议
}

typedef struct objc_method *Methodstruct objc_method{ 
    SEL method_name OBJC2_UNAVAILABLE; // 方法名 char 
    *method_types OBJC2_UNAVAILABLE;
    IMP method_imp OBJC2_UNAVAILABLE; // 方法实现
}

//IMP (Method Implementations) ,IMP 是一个函数指针,这是由编译器生成的,当你发起一个 ObjC 消息之后,最终它会执行的那个代码,就是由这个函数指针指定的
typedef id (*IMP)(id self,SEL _cmd,...); 

//传递消息(编译期其实是在传递消息,并不代表一定被执行)
[target getMovieTitleForObject:obj]; 

//运行期进行消息的分发和转发
 id returnValue = [object messageName:params];
 id returnValue = objc_msgSend(object,@selector(messageName:), params); 

原型为: void  objc_msgSend(id self, SEL cmd, ...)
消息转发机制(可以用来模拟多重继承)

1.动态方法解析:
在对象收到无法解读消息后,首先将调用其所属类的下列类方法:(消息中的选择子为对象方法)

// 返回值表示能否新增一个实例方法处理此选择子
+ (BOOL)resolveInstanceMethod:(SEL)sel;

若选择子为类方法则是 + (BOOL)resolveClassMethod:(SEL)sel
2.备援接受者; 若上述返回为NO, 则继续调用下列方法:

// 如果上述的方法返回为NO则继续执行下列方法
- (id)forwardingTargetForSelector:(SEL)aSelector {
    //能否把这条消息转发为其他接受者处理,是的话将其返回,不是返回nil
    return nil;
}

3.若返回nil则启用完整的消息转发:

- (void)forwardInvocation:(NSInvocation *)anInvocation {
    //创建NSInvocation对象,将消息细节完整封装在其中
    [super forwardInvocation:anInvocation];
}
runtime应用举例:

1.以动态方法解析来实现@dynamic属性,类会自动处理相关属性值得存放与获取操作.细节在 XZSModel 中:

.h 文件
#import <Foundation/Foundation.h>
@interface XZSModel : NSObject
@property (nonatomic, copy) NSString * name;
@property (nonatomic, copy) NSString * area;
@property (nonatomic, copy) NSString * age;
@end
.m 文件
#import "XZSModel.h"
#import <objc/runtime.h>

@interface XZSModel ()
@property (nonatomic, strong) NSMutableDictionary *storeDict;
@end

@implementation XZSModel

@dynamic name,area,age;

- (instancetype)init {
    if (self = [super init]) {
        _storeDict = [[NSMutableDictionary alloc]init];
    }
    return self;
}

//动态方法解析  在对象收到无法解读消息后,首先将调用其所属类的下列类方法:(消息中的选择子为对象方法)
+ (BOOL)resolveInstanceMethod:(SEL)sel {
    NSString *selectorString = NSStringFromSelector(sel);
    if ([selectorString hasPrefix:@"set"]) {
        //set 方法
        class_addMethod(self, sel, (IMP)xzsSetter, "v@:@");
    }else {
        class_addMethod(self, sel, (IMP)xzsGetter, "@@:");
    }
    //其中 “v@:@” 表示返回值和参数,这个符号涉及 Type Encoding,可以参考Apple的文档
    return YES;
}
//set方法
void xzsSetter(id self, SEL _cmd, id value) {
    XZSModel *typeSelf = (XZSModel *)self;
    NSMutableDictionary *storeDict = typeSelf.storeDict;
    
   
    //方法例如: "setName:",因此移除"set"和":",并将处理后的字符的第一个字符改为小写如:"N -> n" 
    NSString *selectorString = NSStringFromSelector(_cmd);
    NSMutableString *key = [selectorString mutableCopy];
    
    [key deleteCharactersInRange:NSMakeRange(key.length - 1, 1)];
    [key deleteCharactersInRange:NSMakeRange(0, 3)];
    
   //第一个字符改为小写
    NSString *lowerFirstChar = [[key substringToIndex:1] lowercaseString];
    [key replaceCharactersInRange:NSMakeRange(0, 1) withString:lowerFirstChar];
    NSLog(@"处理后的key: %@",key);
    
    if (value) {
        [storeDict setValue:value forKey:key];
    }else{
        [storeDict removeObjectForKey:key];
    }
}

// get方法
id xzsGetter(id self, SEL  _cmd){
    XZSModel *typeSelf = (XZSModel *)self;
    NSMutableDictionary *storeDict = typeSelf.storeDict;
    
    NSString *key = NSStringFromSelector(_cmd);
    return [storeDict objectForKey:key];
}

@end

2.利用runtime向类中新增或替换选择子所对应方法的实现

 NSString+XZSAdditions.m 文件
#import "NSString+XZSAdditions.h"

@implementation NSString (XZSAdditions)
 //此方案也称:方法调配(method swizzling)
- (NSString *)xzs_lowercaseSring {
    NSString *lowercase = [self xzs_lowercaseSring];
    NSLog(@"%@ ---> %@",self,lowercase);
    return lowercase;
}

@end
测试验证
    Method originMethod = class_getInstanceMethod([NSString class], @selector(lowercaseString));
    Method newMethod    = class_getInstanceMethod([NSString class], @selector(xzs_lowercaseSring));
    method_exchangeImplementations(originMethod, newMethod);
    
    NSString *string = @"hEllO worLD!";
    NSString *lowercaseString = [string lowercaseString];

控制台打印结果:
hEllO worLD! ---> hello world!

3.利用runtime获取对象所有属性的方法,给模型自动赋值
获取对象的所有成员变量: class_copyIvarList
获取对象的所有属性: class_copyPropertyList

分类 .m 文件
#import "NSObject + model.h"
#import <objc/runtime.h>

#define BY_IVAR 0

@implementation NSObject (model)

+ (instancetype)modelWithDic:(NSDictionary *)dic {
    
    //获取属性列表
    id obejctSelf = [[self alloc]init];
    unsigned int count;
    
#if BY_IVAR
    Ivar * ivarList = class_copyIvarList(self, &count);
#else
    objc_property_t *properties = class_copyPropertyList(self, &count);
#endif
    
    for (int i = 0; i < count; i++) {
#if BY_IVAR
        Ivar ivar = ivarList[i];
        NSString *name = [NSString stringWithUTF8String:ivar_getName(ivar)];
        NSString *type = [NSString stringWithUTF8String:ivar_getTypeEncoding(ivar)];
        NSString *key   = [name substringFromIndex:1];
#else
        objc_property_t property = properties[i];
        NSString *name = [NSString stringWithUTF8String:property_getName(property)];
        NSString *type = [NSString stringWithUTF8String:property_getAttributes(property)];
        NSString *key   = name;
#endif
        
        if ([dic isKindOfClass:[NSNull class]] || dic == nil) {
            continue;
        }
        id value = dic[key];
        
        if ([value isKindOfClass:[NSDictionary class]]) {
            NSRange range = [type rangeOfString:@"\""];
            type = [type substringFromIndex:range.location + range.length];
            range = [type rangeOfString:@"\""];
            type = [type substringToIndex:range.location];
            Class typeClass = NSClassFromString(type);
            if (typeClass) {
                [typeClass modelWithDic:value];
            }
        }
        if (value) {
            [obejctSelf setValue:value forKey:key];
        }
    }
  
    return obejctSelf;
}

@end

4.利用runtime关联对象给分类添加属性

//设置关联对象值 (object:与谁关联,通常是传self;   key:唯一键,在获取值时通过该键获取,通常是使用static,const void *来声明);  value:关联所设置的值;  policy:内存管理策略
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)
typedef OBJC_ENUM(uintptr_t, objc_AssociationPolicy){
OBJC_ASSOCIATION_ASSIGN = 0, // 表示弱引用关联,通常是基本数据类型OBJC_ASSOCIATION_RETAIN_NONATOMIC = 1, // 表示强引用关联对象,是线程安全的
OBJC_ASSOCIATION_COPY_NONATOMIC = 3, // 表示关联对象copy,是线程安全的
OBJC_ASSOCIATION_RETAIN = 01401, // 表示强引用关联对象,不是线程安全的
OBJC_ASSOCIATION_COPY = 01403 // 表示关联对象copy,不是线程安全的
};
//分类添加属性
#import "UIView + changeColor.h"
#import <objc/runtime.h>

static char changeColorChar;

@implementation UIView (changeColor)
@dynamic changeColor;

- (void)setChangeColor:(UIColor *)changeColor {
    objc_setAssociatedObject(self, &changeColorChar, changeColor, OBJC_ASSOCIATION_RETAIN_NONATOMIC);
}

- (UIColor *)changeColor {
    UIColor *color = objc_getAssociatedObject(self, &changeColorChar);
    return color;
}

@end

5.模拟多重继承
待续!

参考资料
1.//www.greatytc.com/p/970ae3bac1ef
2.http://justinyan.me/post/1624
3.52个有效方法
4.Objective-C Runtime Programming Guide

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

推荐阅读更多精彩内容