欢迎来到 IT实训基地-南通科迅教育
咨询电话:0513-81107100
几个有关iOS的几个常见问题-----RunTime
2016/12/10
科迅教育
268
南通零基础IT培训班效果怎么样

说起RunTime这又是一个痛点,学iOS开发有多少是纯粹的因为兴趣?然后又因为兴趣去研究OC。毕竟大学里买的起iPhone,买的起MacBook,买的起开发者账号(xCode7以前如果真机测试是需要开发者账号的)?所以我在这里建议那些面试官,别问你在大学里如何学习的iOS开发的。。。。因为大多数都是像我这样有人带出来的,自学Java遇到不懂的你可以在学校问老师,iOS你丫问谁去?最重要的是iOS火起来是从12年开始的。。。12年你开始干,而且环视国内没啥教程的干。。。10个程序员有3个能学到考虑Runtime这个点就真的很不错了。。。。。所以问这个个人感觉真的没太必要。。。。而且第一批iOS开发者,都是有了四五年开发经验后才开始的iOS开发,你们觉得呢?哪种人真的很少的。。。。好了,吐槽结束。。。。开始我们的重点------Runtime

RunTime

字面上看就是“运行的时候”,说得稍微专业点就是程序在运行时的状态(我在这儿说一下,Runtime是OC语言的专业知识,不是iOS开发的专业知识,而且Java也有Runtime的)。

一、动态 vs 静态语言

OC 是面相运行时的语言,意思是它会尽可能的把编译和链接时要执行的逻辑延迟到运行时。这就给了程序员很大的灵活性,程序员可以根据需要把消息定向给合适的对象,你甚至可以用交换方法的实现(在OC 中调用一个对象的方法可以看成向一个对象发送消息)。这就需要使用 runtime,runtime 可以做对象自省查看他们正在做的和不能做的,并且合适的分发消息(译注:感兴趣的同学可以查看 NSObject 类的 – forwardingTargetForSelector: 和 – forwardInvocation: 方法。P.S. 不是 NSObject 协议! )。如果我们和 C 这样的语言对比。在 C 里,你从 main() 方法开始写然后就是从上到下的写逻辑了并按你写代码的顺序执行程序。一个 C 的结构体不能转发函数执行请求到其他的目标上(other targets)。很可能你的程序是这样的:

 

?
1
 

?
1
2
3
4
5
intmain(intargc,constchar**argv[])
{       
    printf("Hello World!");
    return0;
}
?
1
 
编译器解析,优化然后把优化后的代码转成汇编:
?
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
.text
 .align4,0x90
 .globl _main
_main:
Leh_func_begin1:
 pushq %rbp
Llabel1:
 movq %rsp, %rbp
Llabel2:
 subq $16, %rsp
Llabel3:
 movq %rsi, %rax
 movl %edi, %ecx
 movl %ecx, -8(%rbp)
 movq %rax, -16(%rbp)
 xorb %al, %al
 leaq LC(%rip), %rcx
 movq %rcx, %rdi
 call _printf
 movl $0, -4(%rbp)
 movl -4(%rbp), %eax
 addq $16, %rsp
 popq %rbp
 ret
Leh_func_end1:
 .cstring
LC:
 .asciz"Hello World!"
然后链接库并生成可执行程序要和 OC 对比的话,处理过程很相似,生成的代码依赖于是否有 OC Runtime 库。当刚学 OC 时,我们最先了解的(最简单的那种)是 OC 中用括号包起来的代码像这样…
?
1
[self doSomethingWithVar:var1];
再然后:
?
1
objc_msgSend(self,@selector(doSomethingWithVar:),var1);
之后的么。。。我就找不到是啥了。。。。

二、OC Runtime 是什么?

OC 的 Runtime 是一个运行时库(Runtime Library),它是一个主要使用 C 和汇编写的库,为 C 添加了面相对象的能力并创造了 OC。这就是说它在类信息(Class information) 中被加载,完成所有的方法分发,方法转发,等等。OC runtime 创建了所有需要的结构体,让 OC 的面相对象编程变为可能。

三、OC Runtime 术语

更深入之前,咱们先了解点术语。Mac 和 iPhone 开发者关心的有两个 runtime:Modern Runtime(现代的 Runtime) 和 Legacy Runtime(过时的 Runtime)。Modern Runtime:覆盖所有 64 位的 Mac OS X 应用和所有 iPhone OS 的应用。 Legacy Runtime: 覆盖其他的所有应用(所有 32 位的 Mac OS X 应用) Method 有 2 种基本类型的方法。Instance Method(实例方法):以 ‘-’ 开始,比如 -(void)doFoo; 在对象实例上操作。Class Method(类方法):以 ‘+’ 开始,比如 +(id)alloc。方法(Methods)和 C 的函数很像,是一组代码,执行一个小的任务,如:

?
1
 
- (NSString *)movieTitle { return @"Futurama: Into the Wild Green Yonder"; }
?
1
 
Selector 在 Objective-C 中 selector 只是一个 C 的数据结构,用于表示一个你想在一个对象上执行的 OC 方法。在 runtime 中的定义像这样…
?
1
typedef struct objc_selector  *SEL;
像这样使用…
?
1
SEL aSel =@selector(movieTitle);
Message(消息)
?
1
[target getMovieTitleForObject:obj];
消息是方括号 ‘[]’ 中的那部分,由你要向其发送消息的对象(target),你想要在上面执行的方法(method)还有你发送的参数(arguments)组成。 OC 的消息和 C 函数调用是不同的。事实上,你向一个对象发送消息并不意味着它会执行它。Object(对象)会检查消息的发送者,基于这点再决定是执行一个不同的方法还是转发消息到另一个目标对象上。Class 如果你查看一个类的runtime信息,你会看到这个…
?
1
2
3
4
typedef struct objc_class *Class;
typedef struct objc_object {
    Class isa;
} *id;
这里有几个事情。我们有一个 OC 类的结构体和一个对象的结构体。objc_object 只有一个指向类的 isa 指针,就是我们说的术语 “isa pointer”(isa 指针)。这个 isa 指针是当你向对象发送消息时,OC Runtime 检查一个对象并且查看它的类是什么然后开始查看它是否响应这些 selectors 所需要的一切。最后我么看到了 id 指针。默认情况下 id 指针除了告诉我们它们是 OC 对象外没有其他用了。当你有一个 id 指针,然后你就可以问这个对象是什么类的,看看它是否响应一个方法,等等,然后你就可以在知道这个指针指向的是什么对象后执行更多的操作了。你可以在 LLVM/Clang 的文档中的 Block 中看到
?
1
2
3
4
5
6
7
8
9
10
11
12
13
14
struct Block_literal_1 {
    void*isa;// initialized to &_NSConcreteStackBlock or &_NSConcreteGlobalBlock   
    intflags;   
    intreserved;    
    void(*invoke)(void*, ...);
    struct Block_descriptor_1 {
        unsignedlongintreserved;// NULL    
        unsignedlongintsize; // sizeof(struct Block_literal_1)
        // optional helper functions    
        void(*copy_helper)(void*dst,void*src);
        void(*dispose_helper)(void*src);    
    } *descriptor;   
    // imported variables
};
Blocks 被设计为兼容 OC 的 runtime,所以他们被作为对象对待,因此他们可以响应消息,比如 -retain,-release,-copy ,等等。

IMP(方法实现 Method Implementations)

?
1
typedef id (*IMP)(id self,SEL _cmd,...);
IMP 是指向方法实现的函数指针,由编译器为你生成。如果你新接触 OC 你现在不需要直接接触这些,但是我们将会看到,OC runtime 将如何调用你的方法的。OC Classes(OC 类) 那么什么是 OC 类?在 OC 中的一个类实现看起来像这样:
?
1
 
@interface MyClass : NSObject { // vars NSInteger counter; } // methods -(void)doFoo; @end

但是 runtime 不只要追踪这些
?
1
2
3
4
5
6
7
8
9
10
11
#if!__OBJC2__  
    Class super_class                        OBJC2_UNAVAILABLE;
    constchar*name                         OBJC2_UNAVAILABLE;
    longversion                             OBJC2_UNAVAILABLE;
    longinfo                                OBJC2_UNAVAILABLE;
    longinstance_size                       OBJC2_UNAVAILABLE;   
    struct objc_ivar_list *ivars             OBJC2_UNAVAILABLE;   
    struct objc_method_list **methodLists    OBJC2_UNAVAILABLE;   
    struct objc_cache *cache                 OBJC2_UNAVAILABLE;   
    struct objc_protocol_*protocols          OBJC2_UNAVAILABLE;
#endif

我们可以看到,一个类有其父类的引用,它的名字,实例变量,方法,缓存还有它遵循的协议。runtime 在响应类或实例的方法时需要这些信息。

四、那么 Class 定义的是对象还是对象本身?它是如何实现的(译注:读者需要区分 Class 和 class 是不同的,正如 Nil 和 nil 的用途是不同的)

是的,之前我说过 OC 类也是对象,runtime 通过创建 Meta Classes 来处理这些。当你发送一个消息像这样 [NSObject alloc] 你正在向类对象发送一个消息,这个类对象需要是 MetaClass 的实例,MetaClass 也是 root meta class 的实例。当你说继承自 NSObject 时,你的类指向 NSObject 作为自己的 superclass。然而,所有的 meta class 指向 root metaclass 作为自己的 superclass。所有的 meta class 只是简单的有一个自己响应的方法列表。所以当你向一个类对象发送消息如 [NSObject alloc],然后实际上 objc_msgSend() 会检查 meta class 看看它是否响应这个方法,如果他找到了一个方法,就在这个 Class 对象上执行(译注:class 是一个实例对象的类型,Class 是一个类(class)的类型。对于完全的 OO 来说,类也是个对象,类是类类型(MetaClass)的实例,所以类的类型描述就是 meta class)。

五、为什么我们继承自苹果的类

从你开始 Cocoa 开发时,那些教程就说如继承自 NSObject 然后开始写一些代码,你享受了很多继承自苹果的类所带来的便利。有一件事你从未意识到的是你的对象被设置为使用 OC 的 runtime。当我们为我们的类的一个实例分配了内存,像这样…

?
1
 
MyObject *object = [[MyObject alloc] init];
?
1
 
最先执行的消息是 +alloc。如果你查看下文档, 它说“新的实例对象的 isa 实例变量被初始化为指向一个数据结构,那个数据结构描述了这个类;其他的实例变量被初始化为 0。”所以继承自苹果的类不仅仅是继承了一些重要的属性,也继承了能在内存中轻松分配内存的能力和在内存中创建满足 runtime 期望的对象结构(设置 isa 指针指向我们的类)。
六、那么 Class Cache 是什么?(objc_cache *cache)

 

当 OC runtime 沿着一个对象的 isa 指针检查时,它会发现一个对象实现了许多的方法。然而你可能只调用其中一小部分的方法,也没有意义每次检查时搜索这个类的分发表(dispatch table)中的所有 selector。所以这个类实现了一个缓存,当你搜索一个类的分发表,并找到合适的 selector 后,就会把它放进缓存中。所以当 objc_msgSend() 在一个类中查找 selector 时会先查找类缓存。有个理论是,当你在一个类上调用了一个消息,你很可能之后还会调用它。所以如果我们考虑到这点,就意味着当我们有个子类继承自 NSObject 叫做 MyObject 并且运行了以下的代码:

 

?
1
2
3
4
5
6
7
8
9
MyObject *obj = [[MyObject alloc] init];
@implementationMyObject
- (id)init {
    if(self = [superinit]) {
        [self setVarA:@”blah”];   
    }
    returnself;
}
@end

(1) [MyObject alloc] 首先被执行。MyObject 没有实现 alloc 方法,所以我们不能在这个类中找到 +alloc 方法,然后沿着 superclass 指针会指向 NSObject。

(2) 我们询问 NSObject 是否响应 +alloc 方法,它可以。+alloc 检查消息的接收者类,是 MyObject,然后分配一块和我们的类同样大小的内存空间,并初始化它的 isa 指针指向 MyObject 类,我们现在有了一个实例对象,最终把类对象的 +alloc 方法加入 NSObject 的类缓存(class cache)中(lastly we put +alloc in NSObject's class cache for the class object )。

(3) 到现在为止,我们发送了一个类消息,但是现在我们发送一个实例消息,只是简单的调用 -init 或者我们设计的初始化方法。当然,我们的类会响应这个方法,所以 -(id)init 加入到缓存中。(译注:要是 MyObject 实现了 init 方法,就会把 init 方法加入到 MyObject 的 class cache 中,要是没有实现,只是因为继承才有了这个方法,init 方法还是会加入到 NSObject 的 class cache 中)。

(4) 然后 self = [super init] 被调用。super 是个 magic keyword,指向对象的父类,所以我们得到了 NSObject 并调用它的的 init 方法。这样可以确保 OOP(面相对象编程) 的继承功能正常,这个方法可以正确的初始化父类的变量,之后你(在子类中)可以初始化自己的变量,如果需要可以覆盖父类的方法。在 NSObject 的例子中,没什么重要的要做,但并不总是这样。有时要做些重要的初始化。比如…

?
1
 
#import @interface MyObject : NSObject { NSString *aString; } @property(retain) NSString *aString; @end @implementation MyObject -(id)init { if (self = [super init]) { [self setAString:nil]; } return self; } @synthesize aString; @end int main (int argc, const char * argv[]) { NSAutoreleasePool * pool = [[NSAutoreleasePool alloc] init]; id obj1 = [NSMutableArray alloc]; id obj2 = [[NSMutableArray alloc] init]; id obj3 = [NSArray alloc]; id obj4 = [[NSArray alloc] initWithObjects:@"Hello",nil]; NSLog(@"obj1 class is %@",NSStringFromClass([obj1 class])); NSLog(@"obj2 class is %@",NSStringFromClass([obj2 class])); NSLog(@"obj3 class is %@",NSStringFromClass([obj3 class])); NSLog(@"obj4 class is %@",NSStringFromClass([obj4 class])); id obj5 = [MyObject alloc]; id obj6 = [[MyObject alloc] init]; NSLog(@"obj5 class is %@",NSStringFromClass([obj5 class])); NSLog(@"obj6 class is %@",NSStringFromClass([obj6 class])); [pool drain]; return 0; }
?
1
 
现在如果你新接触 Cocoa ,我让你猜会会输出什么,你可能会说:
?
1
2
3
4
5
6
NSMutableArray
NSMutableArray
NSArray
NSArray
MyObject
MyObject
但实际上是:
?
1
 
obj1 class is __NSPlaceholderArray obj2 class is NSCFArray obj3 class is __NSPlaceholderArray obj4 class is NSCFArray obj5 class is MyObject obj6 class is MyObject
?
1
 

这是因为在 Objective-C 中 +alloc 方法可能会返回某个类的对象,然后在 -init 中返回另一个类的对象。
(译注:感兴趣的同学可以看下这两篇文章:Class Clusters,Make Your Own Abstract Factory Class Cluster in Objective-C, 第二篇文章需要自备小梯子。)

七、那么在 objc_msgSend 中发生了什么?
事实上在 objc_msgSend() 中发生了许多事儿。假设我们有这样的代码…

?
1
 
[self printMessageWithString:@"Hello World!"];
?
1
 
它实际上会被编译器翻译为…
?
1
objc_msgSend(self,@selector(printMessageWithString:),@"Hello World!");

我们沿着目标对象的 isa 指针查找,看看是否这个对象响应 @selector(printMessageWithString:) selector。假设我们在类的分发表或者缓存中找到了这个 selector,我们沿着函数指针并且执行它。这样 objcmsgSend() 就永远不会返回,它开始执行,然后沿着指向方法的指针,然后你的方法返回,这样看起来 objcmsgSend() 方法返回了。Bill Bumgarner 比我讲了更多 objc_msgSend() 的细节(部分1部分2部分3)。

概括下他说的,并且你已经看过了 Objective-C 的 runtime 代码…

检查忽略的 Selector 和短路(Short Circut)—— 显然,如果我们运行在垃圾回收机制下,我们可以忽略调用 -retain, -release, 等等。

检查 nil 对象(target)。和其他的语言不一样的是,在 OC 中向 nil 发送消息是完全合法的,并且有些原因下你会愿意这么做的。假设我们有个非 nil 的对象,然后我们继续…

然后我们需要在这个类上找到 IMP,所以我们先从 class cache 中找起,如果找到了就沿着指针跳到这个函数。

如果没有在缓存中找到 IMP,然后去查找类的分发表,如果找到了,就沿着指针跳到这个函数。

如果 IMP 没有在缓存和类的分发表中找到,然后我们跳到转发机制。这意味着最终你的代码被编译器转换为 C 函数。你写的方法会像这样…

?
1
-(int)doComputeWithNum:(int)aNum
会被译为:
?
1
intaClass_doComputeWithNum(aClass *self,SEL _cmd,intaNum)
OC Runtime 通过调用(invoking)指向这些方法的函数指针调用你的方法(call your methods)。现在,我要说的是,你不能直接调用这些被翻译的方法,但是 Cocoa 框架提供了获得函数指针的方法…
?
1
2
3
4
5
6
int(computeNum *)(id,SEL,int);
//methodForSelector is COCOA & not ObjC Runtime
//gets the same function pointer objc_msgSend gets
computeNum = (int(*)(id,SEL,int))[target methodForSelector:@selector(    doComputeWithNum:)];
//execute the C function pointer returned by the runtime
computeNum(obj,@selector(doComputeWithNum:),aNum);
通过这种方法,你可以直接访问这个函数,并且可以在运行时直接调用,甚至可以使用这个避开 runtime 的动态特性,如果你绝对需要确保一个方法被执行。Objective-C 就是用这种途径去调用你的方法的,但是使用的是 objc_msgSend()。

八、Objective-C 消息转发
在 Objective-C 中向一个不知道如何响应这个方法的对象发送消息是完全合法的(甚至可能是一种潜在的设计决定)。苹果的文档中给出的一个原因是模拟多继 承,Objective-C 不是原生支持的,或者你可能只是想抽象你的设计并且隐藏幕后处理这些消息的其他对象/类。这一点是 runtime 非常需要的。它是这样做的

1. Runtime 检查了你的类和所有父类的 class cache 和分发表,但是没找到指定的方法。

2. OC 的 Runtime 会在你的类上调用 + (BOOL) resolveInstanceMethod:(SEL)aSEL。 这就给了你一个机会去提供一个方法实现并且告诉 runtime 你已经解析了这个方法,如果它开始查找,这回就会找到这个方法。你可以像这样实现…定义一个函数…

?
1
2
3
4
voidfooMethod(id obj, SEL _cmd)
{
    NSLog(@"Doing Foo");
}
然后你可以像这样使用 class_addMethod() 解析它…
?
1
2
3
4
5
6
7
8
9
+(BOOL)resolveInstanceMethod:(SEL)aSEL
{
    if(aSEL ==@selector(doFoo:))
    {
            class_addMethod([selfclass],aSEL,(IMP)fooMethod,"v@:");
            returnYES;
    }
    return[superresolveInstanceMethod];
}
在 class_addMethod() 最后一部分的 "v@:" 是方法的返回和参数类型。你可以在 Runtime Guide 的 Type Encoding 章节看到完整介绍。

3. Runtime 然后调用 – (id)forwardingTargetForSelector:(SEL)aSelector。这样做是为了给你一次机会(因为我们不能解析这个方法 (参见上面的 #2))引导 Objective-C runtime 到另一个可以响应这个消息的对象上,在花费昂贵的处理过程调用 – (void)forwardInvocation:(NSInvocation *)anInvocation 之前调用这个方法也是更好的。你可以像这样实现

?
1
2
3
4
5
6
7
8
- (id)forwardingTargetForSelector:(SEL)aSelector
{
    if(aSelector ==@selector(mysteriousMethod:))
    {       
        returnalternateObject;
    }
    return[superforwardingTargetForSelector:aSelector];
}
显然你不想从这个方法直接返回 self,否则可能会产生一个死循环。

4. Runtime 最后一次会尝试在目标对象上调用 – (void)forwardInvocation:(NSInvocation *)anInvocation。如果你从没看过 NSInvocation,它是 Objective-C 消息的对象形式。一旦你有了一个 NSInvocation 你可以改变这个消息的一切,包括目标对象,selector 和参数。所以你可以这样做…

?
1
2
3
4
5
6
7
8
9
-(void)forwardInvocation:(NSInvocation *)invocation
    SEL invSEL = invocation.selector;   
    if([altObject respondsToSelector:invSEL]) {       
        [invocation invokeWithTarget:altObject];   
    }else{       
        [self doesNotRecognizeSelector:invSEL];   
    }
}
如果你继承自 NSObject,默认它的 – (void)forwardInvocation:(NSInvocation *)anInvocation 实现只是简单的调用 -doesNotRecognizeSelector:,你可以在最后一次机会里覆盖这个方法去做一些事情。(译注:对这块内容有兴趣的同学可以参见:http://www.cnblogs.com/biosli/p/NSObjectinherit2.html
九、Non Fragile ivars(Modern Runtime)(非脆弱的 ivar)

我们最近在 Modern Runtime 里得到的是 Non Fragile ivars 的概念。当编译你的类时,编译器生成了一个 ivar 布局,显示了在你的类中从哪可以访问你的 ivars,获取指向你的对象的指针,查看 ivar 与对象起始字节的偏移关系,和获取读入的变量类型的总共字节大小等一些底层的细节。所以你的 ivar 布局可能看起来像这样,左侧的数字是字节偏移量。

\ 我们有了 NSObject 的 ivar 布局,然后我们继承自 NSObject 去扩展它并且添加了我们自己的 ivars。在苹果发布更新前这都工作的很好,但是 Mac OS X 10.6 发布后,就成了这样

 

\

你的自定义对象被剔除了因为我们有了一个重叠的父类。唯一可以防止这个的办法是如果苹果坚持之前的布局,如果他们这么做了,那么他们的框架就不能改进,因 为他们的 ivar 布局被冻住了。在 fragile ivar 下你不得不重新编译你继承自苹果类的类来恢复兼容性。所以在非 fragile ivar 时,会发生生么?

\

使用非 fragile ivars 时,编译器生成和 fragile ivars 相同的 ivar 布局。然而当 runtime 检测到一个重叠的超类时,它调整你在这个类中新增的 ivar 的偏移量,这样在子类中新增加的那部分就显示出来了。

十、Objective-C 关联对象

最近在 Mac OS X 10.6 雪豹 中新引入了关联引用。Objective-C 不能动态的添加一些属性到对象上,和其他的一些原生支持这点的语言不一样。所以之前你都不得不努力为未来要增加的变量预留好空间。在 Mac OS X 10.6 中,Objective-C 的 Runtime 已经原生的支持这个功能了。如果我们想向一个已有的类添加变量,看起来像这样…

?
1
 
#import //Cocoa #include //objc runtime api’s @interface NSView (CustomAdditions) @property(retain) NSImage *customImage; @end @implementation NSView (CustomAdditions) static char img_key; //has a unique address (identifier) - (NSImage *)customImage { return objc_getAssociatedObject(self,&img_key); } - (void)setCustomImage:(NSImage *)image { objc_setAssociatedObject(self, &img_key,image, OBJC_ASSOCIATION_RETAIN); } @end objc_setAssociatedObject() 的选项,你可以在 runtime.h 文件中找到。 /* Associated Object support. */ /* objc_setAssociatedObject() options */ enum { OBJC_ASSOCIATION_ASSIGN = 0, OBJC_ASSOCIATION_RETAIN_NONATOMIC = 1, OBJC_ASSOCIATION_COPY_NONATOMIC = 3, OBJC_ASSOCIATION_RETAIN = 01401, OBJC_ASSOCIATION_COPY = 01403 };

这些和 @property 语法中的选项意思一样。

混和的 vTable Dispatch

如果你看过 modern runtime 的代码,你会发现这个(在objc-runtime-new.m中)

?
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
/***********************************************************************
* vtable dispatch
*
* Every class gets a vtable pointer. The vtable is an array of IMPs.
* The selectors represented in the vtable are the same for all classes
*   (i.e. no class has a bigger or smaller vtable).
* Each vtable index has an associated trampoline which dispatches to
*   the IMP at that index for the receiver class's vtable (after
*   checking for NULL). Dispatch fixup uses these trampolines instead
*   of objc_msgSend.
* Fragility: The vtable size and list of selectors is chosen at launch
*   time. No compiler-generated code depends on any particular vtable
*   configuration, or even the use of vtable dispatch at all.
* Memory size: If a class's vtable is identical to its superclass's
*   (i.e. the class overrides none of the vtable selectors), then
*   the class points directly to its superclass's vtable. This means
*   selectors to be included in the vtable should be chosen so they are
*   (1) frequently called, but (2) not too frequently overridden. In
*   particular, -dealloc is a bad choice.
* Forwarding: If a class doesn't implement some vtable selector, that
*   selector's IMP is set to objc_msgSend in that class's vtable.
* +initialize: Each class keeps the default vtable (which always
*   redirects to objc_msgSend) until its +initialize is completed.
*   Otherwise, the first message to a class could be a vtable dispatch,
*   and the vtable trampoline doesn't include +initialize checking.
* Changes: Categories, addMethod, and setImplementation all force vtable
*   reconstruction for the class and all of its subclasses, if the
*   vtable selectors are affected.
**********************************************************************/

背后的思想是,runtime 尝试在这个 vtable 中存储最近被调用的 selectors,这样就可以提升你的应用的速度,因为它使用了比 objc_msgSend 更少的指令(fewer instructions)。vtable 中保存 16 个全局最经常调用的 selectors,事实上顺着代码往下看你可以发现垃圾回收和非垃圾回收类型程序的默认 selectors :

 

?
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
staticconstchar*constdefaultVtable[] = {
    "allocWithZone:",
    "alloc",
    "class",
    "self",
    "isKindOfClass:",
    "respondsToSelector:",
    "isFlipped",
    "length",
    "objectForKey:",
    "count",
    "objectAtIndex:",
    "isEqualToString:",
    "isEqual:",
    "retain",
    "release",
    "autorelease",
};
staticconstchar*constdefaultVtableGC[] = {
    "allocWithZone:",
    "alloc",
    "class",
    "self",
    "isKindOfClass:",
    "respondsToSelector:",
    "isFlipped",
    "length",
    "objectForKey:",
    "count",
    "objectAtIndex:",
    "isEqualToString:",
    "isEqual:",
    "hash",
    "addObject:",
    "countByEnumeratingWithState:objects:count:",
};
你可以在调试时从堆栈追踪里找到其中的method,可以像objc_msgSend()一样将它们用于调试。
总结
OC Runtime是非常优秀的作品,它为支撑我们的Cocoa/Objective-C app以及众多的优秀特性做了大量工作。你可以查看苹果官方文档来继续深入了解(Objective-C Runtime Programming GuideObjective-C Runtime Reference)。

最后说几句,各位程序员们,上面的那些大多数部分是别人写的,选择原创是因为我找不到原创作者。。。。所以原创作者看到了的话不要说我盗用,给我留言,我会把出处标在最上方的。。。。还有真想学明白OC下的Runtime,我介意大家买一本叫《Effective Objective - C 2.0》的书。很不错。。。。

77
关闭
先学习,后交费申请表
每期5位名额
在线咨询
免费电话
QQ联系
先学习,后交费
TOP
您好,您想咨询哪门课程呢?
关于我们
机构简介
官方资讯
地理位置
联系我们
0513-91107100
周一至周六     8:30-21:00
微信扫我送教程
手机端访问
南通科迅教育信息咨询有限公司     苏ICP备15009282号     联系地址:江苏省南通市人民中路23-6号新亚大厦三楼             法律顾问:江苏瑞慈律师事务所     Copyright 2008-