runtime详解内容补充

最近研究了一下oc底层的runtime机制,在网上找到了一篇不错的文章对于runtime讲的也比较详细(iOS Runtime详解)。
对于runtime不太了解的同学可以先看下这篇文章,下面是一些我的理解和内容的补充。

关于元类

根据基类NSObject我们知道在每个类中都有一个isa,实际上是一个objc_class的结构体(runtime.h中有显示)。
其中有两个字段是指向其元类的isa和指向父类的supperclass(两者都是Class也就是objc_class)。


结构关系

在objc_class中有一个methodLists字段存放的是类的方法,通过实验(class_copyMethodList获取这个方法列表),我发现类中的methodLists都是实例方法,而元类都是类方法。
所以元类是存储了这个类的类方法,并且保证了子类也能调用到父类的类方法(因为类的元类的父类指向的是类的父类的元类)。

关于消息转发机制的应用(交换两个方法)

这边大多数主要是为了全局修改某个方法实现。
第一种方法,比如给NSMutableDictionary的setObject:forKey:方法添加nil判断,建一个NSMutableDictionary的分类添加下面方法:

+ (void)load {
    [self changeMehtod];
}
+ (void)changeMehtod {
    //这边注意不用[NSMutableDictionary class]是因为NSMutableDictionary属于类簇
    NSMutableDictionary *dict = [NSMutableDictionary new];
    Method originMethod = class_getInstanceMethod(object_getClass(dict), @selector(setObject:forKey:));
    Method targetMethod = class_getInstanceMethod(object_getClass(dict), @selector(handleSetObject:forKey:));
    method_exchangeImplementations(originMethod, targetMethod);
}

- (void)handleSetObject:(id)anobject forKey:(id)akey {
    if (anobject == nil) {
        anobject = @"error";
    }
    //因为交换了方法的关系,系统的setObject:forKey:
    [self handleSetObject:anobject forKey:akey]; 
}

还有通过block的方式,比如给UIViewController要在执行ViewDidLoad前执行一些操作:

+ (void)load {
    [self changeMehtod];
}
+ (void)changeMehtod{
   Method method = class_getInstanceMethod([UIViewController class], @selector(viewDidLoad));
    
    void (*originIMP)(id self, SEL _cmd) = nil;
    originIMP = method_getImplementation(method);
    
    id newViewDidLoad = ^(id self, SEL _cmd){
        NSLog(@"operation");
        originIMP(self, _cmd);
    };
    
    IMP newIMP = imp_implementationWithBlock(newViewDidLoad);
    method_setImplementation(method, newIMP);
}

我们知道oc的方法调用本质上就是消息转发
在一个方法Method中有一个SEL和IMP,SEL是方法的编号实际上就是一个char类型数组字符串,比如viewDidLoad方法的SEL打印出来就是"viewDidLoad"这个在没运行程序时就能确定,而动态变的是IMP是方法的具体实现,还有一个SEL和IMP的对应表。方法交换本质上就是交换SEL和IMP的对应关系。

- (void)methodIMP{
//相当于[self testMethod:@"123"];
    ((void (*)(id ,SEL, NSString*))objc_msgSend)(self, @selector(testMethod:), @"123");
}

- (void)testMethod:(NSString*)agr{  
    NSLog(@"%@", agr);
}

关于分类

对于分类为什么能添加方法或者说覆盖原来类中方法名相同的方法呢?
我做了一个实验,新建一个方法,写了一个方法hello,并建了一个它的分类也实现了这个,方法然后实例话类调用方法,这时候调用的分类的方法。
然后我查看了这个类的方法列表methodLists,发现有两个名字都是hello方法。

    [[TestFunctionOne new] hello];

    int outCount;
    Method *methods = class_copyMethodList([TestFunctionOne class], &outCount);
    for (int i = 0; i < outCount; i++) {
        Method method = methods[i];
        NSLog(@"%s",sel_getName(method_getName(method)));
    }

从这里可以看出category实际上就是把方法插到了类中的methodLists中的前面,在调用方法转发消息的时候,类通过methodLists依次寻找方法,结果先找到了分类的方法。

关于类的动态创建

由于runtime机制的动态性,我们可以通过在代码运行时创建一个类(虽然具体使用场景我没有想到..),具体如下:

+ (void)createClass{
    //TestClass判断这个类是否存在
    if (NSClassFromString(@"TestClass")) {
        return;
    }
    
    // 1 创建
    Class EOCTestClass = objc_allocateClassPair([NSObject class], "TestClass", 0);
    
    // 2 添加变量
    if(class_addIvar(TestClass, "str", sizeof(NSString*), log2(sizeof(NSString*)), @encode(NSString*))){
        NSLog(@"添加成功");
    }
    
    if(class_addIvar(EOCTestClass, "intNum", sizeof(int), log2(sizeof(int)), @encode(int))){
        NSLog(@"添加成功");
    }
    
    // 3 提交 (提交完之后, 就不能在添加变量)
    objc_registerClassPair(TestClass);
    
    // 4 测试
    id testObject = [TestClass new];
    
    [testObject setValue:@"string" forKey:@"str"];
    NSLog(@"%@", [testObject valueForKey:@"str"]);    
}

就先写这些,之后有新的体会在补充 :)

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

推荐阅读更多精彩内容

  • 转至元数据结尾创建: 董潇伟,最新修改于: 十二月 23, 2016 转至元数据起始第一章:isa和Class一....
    40c0490e5268阅读 1,690评论 0 9
  • 引导 对于从事 iOS 开发人员来说,所有的人都会答出「 Runtime 是运行时 」,什么情况下用 Runtim...
    Winny_园球阅读 4,197评论 3 75
  • 这篇文章完全是基于南峰子老师博客的转载 这篇文章完全是基于南峰子老师博客的转载 这篇文章完全是基于南峰子老师博客的...
    西木阅读 30,548评论 33 466
  • 对于从事 iOS 开发人员来说,所有的人都会答出【runtime 是运行时】什么情况下用runtime?大部分人能...
    梦夜繁星阅读 3,700评论 7 64
  • 我们常常会听说 Objective-C 是一门动态语言,那么这个「动态」表现在哪呢?我想最主要的表现就是 Obje...
    Ethan_Struggle阅读 2,186评论 0 7