重温Runtime

刚刚接触到,犹如窥探到一座宝藏,深深被他所以吸引.感觉能用它做很多牛逼的事,但是具体能做什么却又不知道. 可能这就是接触新东西所带给我的体验,直觉告诉它确实有无限的创造力.

Objective-C 调用方法,每天被用到无数次,写了无数遍的alloc init,从来没有想过这个到底怎么调用.今天撑着兴致高涨,做点学习笔记.

首先,OC调用一个类的方法,并不是把这个方法编译成一个唯一的符号,然后调用的时候查找符号表. OC在runtime层会走:voidobjc_msgSend(void/* id self, SEL op, ... */) 这样一个方法.如注释所写,参数分别是方法调用者,方法选择器,方法参数. 那么msgSend又是如何做的分发呢,查询源码注释可知:

判断receiver是否为nil,也就是objc_msgSend的第一个参数self,也就是要调用的那个方法所属对象从缓存里寻找,找到了则分发,否则利用objc-class.mm中_class_lookupMethodAndLoadCache3方法去寻找selector

从本class的method list寻找selector,如果找到,填充到缓存中,并返回selector,否则寻找父类的method list,并依次往上寻找,直到找到selector,填充到缓存中,并返回selector,否则调用_class_resolveMethod,如果可以动态resolve为一个selector,不缓存,方法返回,否则转发这个selector

else 抛出异常.

简单总结;

首先,通过 obj 的 isa 指针找到它的 class ;

在 class 的 method list 找 foo ;

如果 class 中没到 foo,继续往它的 superclass 中找 ;

一旦找到 foo 这个函数,就去执行它的实现IMP


我们在实际调用中不难发现,当我们用一个下层类去调用上层类的每一个方法时如果没有缓存,那么整个查找链是相当长的.就算方法是在这个类里面,当方法比较多的时候,每次都查找也是费事费力的一件事情.当一个方法重复被调用,哪怕只重复一次,那做缓存都是相当有必要的.


让我们来看一下objc_cache到底是什么:

struct objc_cache {

uintptr_t mask;

uintprt_t occupied;

cache_entry *buckets[1];

}

objc_cache的定义看起来很简单,它包含了下面三个变量:

1)、mask:可以认为是当前能达到的最大index(从0开始的),所以缓存的size(total)是mask+1

2)、occupied:被占用的槽位,因为缓存是以散列表的形式存在的,所以会有空槽,而occupied表示当前被占用的数目

3)、buckets:用数组表示的hash表,cache_entry类型,每一个cache_entry代表一个方法缓存

(buckets定义在objc_cache的最后,说明这是一个可变长度的数组)


cache_entry定义也包含了三个字段,分别是:

typedef struct {

SEL name;

void *unused;

IMP imp;

}

1)、name,被缓存的方法名字

2)、unused,保留字段,还没被使用。

3)、imp,方法实现

缓存的存储使用了散列表。因为散列表检索起来更快,sel被散列后找到一个空槽放在buckets中.

Class 类的缓存存在metaclass中, 所以每个类都只有一份方法缓存,而不是每一个类的object都保存一份. 从父类取到的方法,也会存在类本身的方法缓存里。而当用一个父类对象去调用那个方法的时候,也会在父类的metaclass里缓存一份。

简单总结:

这也就是objc_class中另一个重要成员objc_cache做的事情 - 再找到 foo 之后,把 foo 的method_name作为 key ,method_imp作为 value 给存起来。当再次收到 foo 消息的时候,可以直接在 cache 里找到,避免去遍历objc_method_list.


类方法表是list 有序的,这就是为什么我们在写类别的时候,最好是不要重写系统的方法,其实并不是覆盖掉了系统原有的方法,只是通过runtime增加了一个新的方法,但是这个方法会排在系统原有方法的前面,所以调用的时候,由于list有序,所以优先调用了重写的方法.

我们应该如何调用被"被覆盖的方法"呢

Class currentClass = [MyClass Class];

Class currentClass = [[MVVMalloc]init];

MVVM*my = [[MVVMalloc]init];

if(currentClass) {

unsignedintcount;

Method*methodList =class_copyMethodList(currentClass, &count);

IMPlastimp =nil;

SELlastsel =nil;

for(NSIntegeri =0; i

Methodmethod = methodList[i];

NSString*methodName = [NSStringstringWithCString:sel_getName(method)encoding:NSUTF8StringEncoding];

if([methodNameisEqualToString:@"方法名"]) {

lastimp =method_getImplementation(method);

lastsel =method_getName(method);

}

}

typedefvoid(*fn)(id,SEL);

if(lastimp !=NULL) {

fnf = (fn)lastimp;

f(my,lastsel);

}

free(methodList);

}


当你有一个类有两个类别,并且拥有同一方法的时候,可以通过控制编译顺序选择优先加载那个方法.(Compile Sources 里的文件顺序)

学习源自--美团技术技术团队博客, GLOW技术团队

最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
平台声明:文章内容(如有图片或视频亦包括在内)由作者上传并发布,文章内容仅代表作者本人观点,简书系信息发布平台,仅提供信息存储服务。

推荐阅读更多精彩内容

  • 转至元数据结尾创建: 董潇伟,最新修改于: 十二月 23, 2016 转至元数据起始第一章:isa和Class一....
    40c0490e5268阅读 5,842评论 0 9
  • 我们常常会听说 Objective-C 是一门动态语言,那么这个「动态」表现在哪呢?我想最主要的表现就是 Obje...
    Ethan_Struggle阅读 6,593评论 0 7
  • 转载:http://yulingtianxia.com/blog/2014/11/05/objective-c-r...
    F麦子阅读 4,108评论 0 2
  • 这篇文章完全是基于南峰子老师博客的转载 这篇文章完全是基于南峰子老师博客的转载 这篇文章完全是基于南峰子老师博客的...
    西木阅读 30,717评论 33 466
  • 今天终于等到了向心理咨询师高高老师的语音咨询,这是有生以来第一次和心理咨询师正面接触,虽然就要只有一个小时,但是一...
    珍惜年青阅读 1,293评论 1 1