Objective-C Runtime 運行時之一:類與對象

本文轉載自:http://southpeak.github.io/2014/10/25/objective-c-runtime-1/

Objective-C語言是一門動態語言,它將很多靜態語言在編譯和鏈接時期做的事放到了運行時來處理。這種動態語言的優勢在于:我們寫代碼時更具靈活性,如我們可以把消息轉發給我們想要的對象,或者隨意交換一個方法的實現等。

這種特性意味著Objective-C不僅需要一個編譯器,還需要一個運行時系統來執行編譯的代碼。對于Objective-C來說,這個運行時系統就像一個操作系統一樣:它讓所有的工作可以正常的運行。這個運行時系統即Objc Runtime。Objc Runtime其實是一個Runtime庫,它基本上是用C和匯編寫的,這個庫使得C語言有了面向對象的能力。

Runtime庫主要做下面幾件事:

封裝:在這個庫中,對象可以用C語言中的結構體表示,而方法可以用C函數來實現,另外再加上了一些額外的特性。這些結構體和函數被runtime函數封裝后,我們就可以在程序運行時創建,檢查,修改類、對象和它們的方法了。

找出方法的最終執行代碼:當程序執行[object doSomething]時,會向消息接收者(object)發送一條消息(doSomething),runtime會根據消息接收者是否能響應該消息而做出不同的反應。這將在后面詳細介紹。

Objective-C runtime目前有兩個版本:Modern runtime和Legacy runtime。Modern Runtime覆蓋了64位的Mac OS X Apps,還有iOS Apps,Legacy Runtime是早期用來給32位Mac OS X Apps用的,也就是可以不用管就是了。

在這一系列文章中,我們將介紹runtime的基本工作原理,以及如何利用它讓我們的程序變得更加靈活。在本文中,我們先來介紹一下類與對象,這是面向對象的基礎,我們看看在Runtime中,類是如何實現的。

類與對象基礎數據結構

Class

Objective-C類是由Class類型來表示的,它實際上是一個指向objc_class結構體的指針。它的定義如下:

typedef struct objc_class *Class;

查看objc/runtime.h中objc_class結構體的定義如下:

structobjc_class {

Class isa? OBJC_ISA_AVAILABILITY;

#if !__OBJC2__

Class super_class? ? ? ? ? ? ? ? ? OBJC2_UNAVAILABLE;// 父類

constchar*name? ? ? ? ? ? ? ? ? ? ? OBJC2_UNAVAILABLE;// 類名

longversion? ? ? ? ? ? ? ? ? ? ? ? ? OBJC2_UNAVAILABLE;// 類的版本信息,默認為0

longinfo? ? ? ? ? ? ? ? ? ? ? ? ? ? OBJC2_UNAVAILABLE;// 類信息,供運行期使用的一些位標識

longinstance_size? ? ? ? ? ? ? ? ? OBJC2_UNAVAILABLE;// 該類的實例變量大小

structobjc_ivar_list *ivars? ? ? ? OBJC2_UNAVAILABLE;// 該類的成員變量鏈表

structobjc_method_list **methodLists OBJC2_UNAVAILABLE;// 方法定義的鏈表

structobjc_cache *cache? ? ? ? ? ? ? OBJC2_UNAVAILABLE;// 方法緩存

structobjc_protocol_list *protocols OBJC2_UNAVAILABLE;// 協議鏈表

#endif

} OBJC2_UNAVAILABLE;

在這個定義中,下面幾個字段是我們感興趣的

isa:需要注意的是在Objective-C中,所有的類自身也是一個對象,這個對象的Class里面也有一個isa指針,它指向metaClass(元類),我們會在后面介紹它。

super_class:指向該類的父類,如果該類已經是最頂層的根類(如NSObject或NSProxy),則super_class為NULL。

cache:用于緩存最近使用的方法。一個接收者對象接收到一個消息時,它會根據isa指針去查找能夠響應這個消息的對象。在實際使用中,這個對象只有一部分方法是常用的,很多方法其實很少用或者根本用不上。這種情況下,如果每次消息來時,我們都是methodLists中遍歷一遍,性能勢必很差。這時,cache就派上用場了。在我們每次調用過一個方法后,這個方法就會被緩存到cache列表中,下次調用的時候runtime就會優先去cache中查找,如果cache沒有,才去methodLists中查找方法。這樣,對于那些經常用到的方法的調用,但提高了調用的效率。

version:我們可以使用這個字段來提供類的版本信息。這對于對象的序列化非常有用,它可是讓我們識別出不同類定義版本中實例變量布局的改變。

針對cache,我們用下面例子來說明其執行過程:

NSArray*array = [[NSArrayalloc] init];

```

其流程是:

1.`[NSArrayalloc]`先被執行。因為NSArray沒有`+alloc`方法,于是去父類NSObject去查找。

2.檢測NSObject是否響應`+alloc`方法,發現響應,于是檢測NSArray類,并根據其所需的內存空間大小開始分配內存空間,然后把`isa`指針指向NSArray類。同時,`+alloc`也被加進cache列表里面。

3.接著,執行`-init`方法,如果NSArray響應該方法,則直接將其加入`cache`;如果不響應,則去父類查找。

4.在后期的操作中,如果再以`[[NSArrayalloc] init]`這種方式來創建數組,則會直接從cache中取出相應的方法,直接調用。

### objc_object與id

`objc_object`是表示一個類的實例的結構體,它的定義如下(`objc/objc.h`):

```objc

structobjc_object {

Class isa? OBJC_ISA_AVAILABILITY;

};

typedefstructobjc_object *id;

可以看到,這個結構體只有一個字體,即指向其類的isa指針。這樣,當我們向一個Objective-C對象發送消息時,運行時庫會根據實例對象的isa指針找到這個實例對象所屬的類。Runtime庫會在類的方法列表及父類的方法列表中去尋找與消息對應的selector指向的方法。找到后即運行這個方法。

當創建一個特定類的實例對象時,分配的內存包含一個objc_object數據結構,然后是類的實例變量的數據。NSObject類的alloc和allocWithZone:方法使用函數class_createInstance來創建objc_object數據結構。

另外還有我們常見的id,它是一個objc_object結構類型的指針。它的存在可以讓我們實現類似于C++中泛型的一些操作。該類型的對象可以轉換為任何一種對象,有點類似于C語言中void *指針類型的作用。

objc_cache

上面提到了objc_class結構體中的cache字段,它用于緩存調用過的方法。這個字段是一個指向objc_cache結構體的指針,其定義如下:

struct objc_cache {

unsignedintmask/* total = mask + 1 */OBJC2_UNAVAILABLE;

unsignedintoccupied? ? ? ? ? ? ? ? ? ? ? ? ? ? ? ? ? ? OBJC2_UNAVAILABLE;

Method buckets[1]? ? ? ? ? ? ? ? ? ? ? ? ? ? ? ? ? ? ? ? OBJC2_UNAVAILABLE;

};

該結構體的字段描述如下:

mask:一個整數,指定分配的緩存bucket的總數。在方法查找過程中,Objective-C runtime使用這個字段來確定開始線性查找數組的索引位置。指向方法selector的指針與該字段做一個AND位操作(index = (mask & selector))。這可以作為一個簡單的hash散列算法。

occupied:一個整數,指定實際占用的緩存bucket的總數。

buckets:指向Method數據結構指針的數組。這個數組可能包含不超過mask+1個元素。需要注意的是,指針可能是NULL,表示這個緩存bucket沒有被占用,另外被占用的bucket可能是不連續的。這個數組可能會隨著時間而增長。

元類(Meta Class)

在上面我們提到,所有的類自身也是一個對象,我們可以向這個對象發送消息(即調用類方法)。如:

NSArray*array = [NSArrayarray];

這個例子中,+array消息發送給了NSArray類,而這個NSArray也是一個對象。既然是對象,那么它也是一個objc_object指針,它包含一個指向其類的一個isa指針。那么這些就有一個問題了,這個isa指針指向什么呢?為了調用+array方法,這個類的isa指針必須指向一個包含這些類方法的一個objc_class結構體。這就引出了meta-class的概念

meta-class是一個類對象的類。

當我們向一個對象發送消息時,runtime會在這個對象所屬的這個類的方法列表中查找方法;而向一個類發送消息時,會在這個類的meta-class的方法列表中查找。

meta-class之所以重要,是因為它存儲著一個類的所有類方法。每個類都會有一個單獨的meta-class,因為每個類的類方法基本不可能完全相同。

再深入一下,meta-class也是一個類,也可以向它發送一個消息,那么它的isa又是指向什么呢?為了不讓這種結構無限延伸下去,Objective-C的設計者讓所有的meta-class的isa指向基類的meta-class,以此作為它們的所屬類。即,任何NSObject繼承體系下的meta-class都使用NSObject的meta-class作為自己的所屬類,而基類的meta-class的isa指針是指向它自己。這樣就形成了一個完美的閉環。

通過上面的描述,再加上對objc_class結構體中super_class指針的分析,我們就可以描繪出類及相應meta-class類的一個繼承體系了,如下圖所示:

對于NSObject繼承體系來說,其實例方法對體系中的所有實例、類和meta-class都是有效的;而類方法對于體系內的所有類和meta-class都是有效的。

講了這么多,我們還是來寫個例子吧:

void TestMetaClass(idself, SEL _cmd) {

NSLog(@"This objcet is %p",self);

NSLog(@"Class is %@, super class is %@", [selfclass], [selfsuperclass]);

Class currentClass = [selfclass];

for(inti =0; i <4; i++) {

NSLog(@"Following the isa pointer %d times gives %p", i, currentClass);

currentClass = objc_getClass((__bridgevoid*)currentClass);

}

NSLog(@"NSObject's class is %p", [NSObjectclass]);

NSLog(@"NSObject's meta class is %p", objc_getClass((__bridgevoid*)[NSObjectclass]));

}

#pragma mark -

@implementationTest

- (void)ex_registerClassPair {

Class newClass = objc_allocateClassPair([NSErrorclass],"TestClass",0);

class_addMethod(newClass,@selector(testMetaClass), (IMP)TestMetaClass,"v@:");

objc_registerClassPair(newClass);

idinstance = [[newClass alloc] initWithDomain:@"some domain"code:0userInfo:nil];

[instance performSelector:@selector(testMetaClass)];

}

@end

這個例子是在運行時創建了一個NSError的子類TestClass,然后為這個子類添加一個方法testMetaClass,這個方法的實現是TestMetaClass函數。

運行后,打印結果是

2014-10-20 22:57:07.352 mountain[1303:41490] This objcet is 0x7a6e22b0

2014-10-20 22:57:07.353 mountain[1303:41490] Class is TestStringClass, super class is NSError

2014-10-20 22:57:07.353 mountain[1303:41490] Following the isa pointer 0 times gives 0x7a6e21b0

2014-10-20 22:57:07.353 mountain[1303:41490] Following the isa pointer 1 times gives 0x0

2014-10-20 22:57:07.353 mountain[1303:41490] Following the isa pointer 2 times gives 0x0

2014-10-20 22:57:07.353 mountain[1303:41490] Following the isa pointer 3 times gives 0x0

2014-10-20 22:57:07.353 mountain[1303:41490] NSObject's class is 0xe10000

2014-10-20 22:57:07.354 mountain[1303:41490] NSObject's meta class is 0x0

我們在for循環中,我們通過objc_getClass來獲取對象的isa,并將其打印出來,依此一直回溯到NSObject的meta-class。分析打印結果,可以看到最后指針指向的地址是0x0,即NSObject的meta-class的類地址。

這里需要注意的是:我們在一個類對象調用class方法是無法獲取meta-class,它只是返回類而已。

類與對象操作函數

runtime提供了大量的函數來操作類與對象。類的操作方法大部分是以class_為前綴的,而對象的操作方法大部分是以objc_或object_為前綴。下面我們將根據這些方法的用途來分類討論這些方法的使用。

類相關操作函數

我們可以回過頭去看看objc_class的定義,runtime提供的操作類的方法主要就是針對這個結構體中的各個字段的。下面我們分別介紹這一些的函數。并在最后以實例來演示這些函數的具體用法。

類名(name)

類名操作的函數主要有:

// 獲取類的類名

constchar* class_getName ( Class cls );

對于class_getName函數,如果傳入的cls為Nil,則返回一個字字符串。

父類(super_class)和元類(meta-class)

父類和元類操作的函數主要有:

// 獲取類的父類

Class class_getSuperclass ( Class cls );

// 判斷給定的Class是否是一個元類

BOOLclass_isMetaClass ( Class cls );

class_getSuperclass函數,當cls為Nil或者cls為根類時,返回Nil。不過通常我們可以使用NSObject類的superclass方法來達到同樣的目的。

class_isMetaClass函數,如果是cls是元類,則返回YES;如果否或者傳入的cls為Nil,則返回NO。

實例變量大小(instance_size)

實例變量大小操作的函數有:

// 獲取實例大小

size_t class_getInstanceSize ( Class cls );

成員變量(ivars)及屬性

在objc_class中,所有的成員變量、屬性的信息是放在鏈表ivars中的。ivars是一個數組,數組中每個元素是指向Ivar(變量信息)的指針。runtime提供了豐富的函數來操作這一字段。大體上可以分為以下幾類:

1.成員變量操作函數,主要包含以下函數:

// 獲取類中指定名稱實例成員變量的信息

Ivar class_getInstanceVariable ( Class cls,constchar*name );

// 獲取類成員變量的信息

Ivar class_getClassVariable ( Class cls,constchar*name );

// 添加成員變量

BOOLclass_addIvar ( Class cls,constchar*name, size_t size, uint8_t alignment,constchar*types );

// 獲取整個成員變量列表

Ivar * class_copyIvarList ( Class cls,unsignedint*outCount );

class_getInstanceVariable函數,它返回一個指向包含name指定的成員變量信息的objc_ivar結構體的指針(Ivar)。

class_getClassVariable函數,目前沒有找到關于Objective-C中類變量的信息,一般認為Objective-C不支持類變量。注意,返回的列表不包含父類的成員變量和屬性。

Objective-C不支持往已存在的類中添加實例變量,因此不管是系統庫提供的提供的類,還是我們自定義的類,都無法動態添加成員變量。但如果我們通過運行時來創建一個類的話,又應該如何給它添加成員變量呢?這時我們就可以使用class_addIvar函數了。不過需要注意的是,這個方法只能在objc_allocateClassPair函數與objc_registerClassPair之間調用。另外,這個類也不能是元類。成員變量的按字節最小對齊量是1<

class_copyIvarList函數,它返回一個指向成員變量信息的數組,數組中每個元素是指向該成員變量信息的objc_ivar結構體的指針。這個數組不包含在父類中聲明的變量。outCount指針返回數組的大小。需要注意的是,我們必須使用free()來釋放這個數組。

2.屬性操作函數,主要包含以下函數:

// 獲取指定的屬性

objc_property_t class_getProperty ( Class cls,constchar*name );

// 獲取屬性列表

objc_property_t * class_copyPropertyList ( Class cls,unsignedint*outCount );

// 為類添加屬性

BOOLclass_addProperty ( Class cls,constchar*name,constobjc_property_attribute_t *attributes,unsignedintattributeCount );

// 替換類的屬性

voidclass_replaceProperty ( Class cls,constchar*name,constobjc_property_attribute_t *attributes,unsignedintattributeCount );

這一種方法也是針對ivars來操作,不過只操作那些是屬性的值。我們在后面介紹屬性時會再遇到這些函數。

3.在MAC OS X系統中,我們可以使用垃圾回收器。runtime提供了幾個函數來確定一個對象的內存區域是否可以被垃圾回收器掃描,以處理strong/weak引用。這幾個函數定義如下:

constuint8_t * class_getIvarLayout ( Class cls );

voidclass_setIvarLayout ( Class cls,constuint8_t *layout );

constuint8_t * class_getWeakIvarLayout ( Class cls );

voidclass_setWeakIvarLayout ( Class cls,constuint8_t *layout );

但通常情況下,我們不需要去主動調用這些方法;在調用objc_registerClassPair時,會生成合理的布局。在此不詳細介紹這些函數。

方法(methodLists)

方法操作主要有以下函數:

// 添加方法

BOOLclass_addMethod ( Class cls, SEL name, IMP imp,constchar*types );

// 獲取實例方法

Method class_getInstanceMethod ( Class cls, SEL name );

// 獲取類方法

Method class_getClassMethod ( Class cls, SEL name );

// 獲取所有方法的數組

Method * class_copyMethodList ( Class cls,unsignedint*outCount );

// 替代方法的實現

IMP class_replaceMethod ( Class cls, SEL name, IMP imp,constchar*types );

// 返回方法的具體實現

IMP class_getMethodImplementation ( Class cls, SEL name );

IMP class_getMethodImplementation_stret ( Class cls, SEL name );

// 類實例是否響應指定的selector

BOOLclass_respondsToSelector ( Class cls, SEL sel );

class_addMethod的實現會覆蓋父類的方法實現,但不會取代本類中已存在的實現,如果本類中包含一個同名的實現,則函數會返回NO。如果要修改已存在實現,可以使用method_setImplementation。一個Objective-C方法是一個簡單的C函數,它至少包含兩個參數–self和_cmd。所以,我們的實現函數(IMP參數指向的函數)至少需要兩個參數,如下所示:

void myMethodIMP(idself, SEL _cmd)

{

// implementation ....

}

與成員變量不同的是,我們可以為類動態添加方法,不管這個類是否已存在。

另外,參數types是一個描述傳遞給方法的參數類型的字符數組,這就涉及到類型編碼,我們將在后面介紹。

class_getInstanceMethod、class_getClassMethod函數,與class_copyMethodList不同的是,這兩個函數都會去搜索父類的實現。

class_copyMethodList函數,返回包含所有實例方法的數組,如果需要獲取類方法,則可以使用class_copyMethodList(object_getClass(cls), &count)(一個類的實例方法是定義在元類里面)。該列表不包含父類實現的方法。outCount參數返回方法的個數。在獲取到列表后,我們需要使用free()方法來釋放它。

class_replaceMethod函數,該函數的行為可以分為兩種:如果類中不存在name指定的方法,則類似于class_addMethod函數一樣會添加方法;如果類中已存在name指定的方法,則類似于method_setImplementation一樣替代原方法的實現。

class_getMethodImplementation函數,該函數在向類實例發送消息時會被調用,并返回一個指向方法實現函數的指針。這個函數會比method_getImplementation(class_getInstanceMethod(cls, name))更快。返回的函數指針可能是一個指向runtime內部的函數,而不一定是方法的實際實現。例如,如果類實例無法響應selector,則返回的函數指針將是運行時消息轉發機制的一部分。

class_respondsToSelector函數,我們通常使用NSObject類的respondsToSelector:或instancesRespondToSelector:方法來達到相同目的。

協議(objc_protocol_list)

協議相關的操作包含以下函數:

// 添加協議

BOOLclass_addProtocol ( Class cls, Protocol *protocol );

// 返回類是否實現指定的協議

BOOLclass_conformsToProtocol ( Class cls, Protocol *protocol );

// 返回類實現的協議列表

Protocol * class_copyProtocolList ( Class cls,unsignedint*outCount );

class_conformsToProtocol函數可以使用NSObject類的conformsToProtocol:方法來替代。

class_copyProtocolList函數返回的是一個數組,在使用后我們需要使用free()手動釋放。

版本(version)

版本相關的操作包含以下函數:

// 獲取版本號

intclass_getVersion ( Class cls );

// 設置版本號

voidclass_setVersion ( Class cls,intversion );

其它

runtime還提供了兩個函數來供CoreFoundation的tool-free bridging使用,即:

Class objc_getFutureClass (constchar*name );

voidobjc_setFutureClass ( Class cls,constchar*name );

通常我們不直接使用這兩個函數。

實例(Example)

上面列舉了大量類操作的函數,下面我們寫個實例,來看看這些函數的實例效果:

//-----------------------------------------------------------

// MyClass.h

@interface MyClass:NSObject

@property(nonatomic,strong )NSArray *array;

@property(nonatomic,copy) NSString *string;

- (void)method1;

- (void)method2;

+ (void)classMethod1;

@end

//-----------------------------------------------------------

// MyClass.m

#import"MyClass.h"

@interface MyClass(){

NSInteger_instance1;

NSString*? _instance2;

}

@property(nonatomic,assign) NSUInteger integer;

- (void)method3WithArg1:(NSInteger)arg1 arg2:(NSString*)arg2;

@end

@implementationMyClass

+ (void)classMethod1 {

}

- (void)method1 {

NSLog(@"call method method1");

}

- (void)method2 {

}

- (void)method3WithArg1:(NSInteger)arg1 arg2:(NSString*)arg2 {

NSLog(@"arg1 : %ld, arg2 : %@", arg1, arg2);

}

@end

//-----------------------------------------------------------

// main.h

#import"MyClass.h"

#import"MySubClass.h"

#import

int main(int argc,const char * argv[]) {

@autoreleasepool{

MyClass *myClass = [[MyClass alloc] init];

unsignedintoutCount =0;

Class cls = myClass.class;

// 類名

NSLog(@"class name: %s", class_getName(cls));

NSLog(@"==========================================================");

// 父類

NSLog(@"super class name: %s", class_getName(class_getSuperclass(cls)));

NSLog(@"==========================================================");

// 是否是元類

NSLog(@"MyClass is %@ a meta-class", (class_isMetaClass(cls) ?@"":@"not"));

NSLog(@"==========================================================");

Class meta_class = objc_getMetaClass(class_getName(cls));

NSLog(@"%s's meta-class is %s", class_getName(cls), class_getName(meta_class));

NSLog(@"==========================================================");

// 變量實例大小

NSLog(@"instance size: %zu", class_getInstanceSize(cls));

NSLog(@"==========================================================");

// 成員變量

Ivar *ivars = class_copyIvarList(cls, &outCount);

for(inti =0; i < outCount; i++) {

Ivar ivar = ivars[i];

NSLog(@"instance variable's name: %s at index: %d", ivar_getName(ivar), i);

}

free(ivars);

Ivar string = class_getInstanceVariable(cls,"_string");

if(string !=NULL) {

NSLog(@"instace variable %s", ivar_getName(string));

}

NSLog(@"==========================================================");

// 屬性操作

objc_property_t * properties = class_copyPropertyList(cls, &outCount);

for(inti =0; i < outCount; i++) {

objc_property_t property = properties[i];

NSLog(@"property's name: %s", property_getName(property));

}

free(properties);

objc_property_t array = class_getProperty(cls,"array");

if(array !=NULL) {

NSLog(@"property %s", property_getName(array));

}

NSLog(@"==========================================================");

// 方法操作

Method *methods = class_copyMethodList(cls, &outCount);

for(inti =0; i < outCount; i++) {

Method method = methods[i];

NSLog(@"method's signature: %s", method_getName(method));

}

free(methods);

Method method1 = class_getInstanceMethod(cls,@selector(method1));

if(method1 !=NULL) {

NSLog(@"method %s", method_getName(method1));

}

Method classMethod = class_getClassMethod(cls,@selector(classMethod1));

if(classMethod !=NULL) {

NSLog(@"class method : %s", method_getName(classMethod));

}

NSLog(@"MyClass is%@ responsd to selector: method3WithArg1:arg2:", class_respondsToSelector(cls,@selector(method3WithArg1:arg2:)) ?@"":@" not");

IMP imp = class_getMethodImplementation(cls,@selector(method1));

imp();

NSLog(@"==========================================================");

// 協議

Protocol * __unsafe_unretained* protocols = class_copyProtocolList(cls, &outCount);

Protocol * protocol;

for(inti =0; i < outCount; i++) {

protocol = protocols[i];

NSLog(@"protocol name: %s", protocol_getName(protocol));

}

NSLog(@"MyClass is%@ responsed to protocol %s", class_conformsToProtocol(cls, protocol) ?@"":@" not", protocol_getName(protocol));

NSLog(@"==========================================================");

}

return0;

}

這段程序的輸出如下:

2014-10-22 19:41:37.452 RuntimeTest[3189:156810] class name: MyClass

2014-10-22 19:41:37.453 RuntimeTest[3189:156810] ==========================================================

2014-10-22 19:41:37.454 RuntimeTest[3189:156810] super class name: NSObject

2014-10-22 19:41:37.454 RuntimeTest[3189:156810] ==========================================================

2014-10-22 19:41:37.454 RuntimeTest[3189:156810] MyClass is not a meta-class

2014-10-22 19:41:37.454 RuntimeTest[3189:156810] ==========================================================

2014-10-22 19:41:37.454 RuntimeTest[3189:156810] MyClass's meta-class is MyClass

2014-10-22 19:41:37.455 RuntimeTest[3189:156810] ==========================================================

2014-10-22 19:41:37.455 RuntimeTest[3189:156810] instance size: 48

2014-10-22 19:41:37.455 RuntimeTest[3189:156810] ==========================================================

2014-10-22 19:41:37.455 RuntimeTest[3189:156810] instance variable's name: _instance1 at index: 0

2014-10-22 19:41:37.455 RuntimeTest[3189:156810] instance variable's name: _instance2 at index: 1

2014-10-22 19:41:37.455 RuntimeTest[3189:156810] instance variable's name: _array at index: 2

2014-10-22 19:41:37.455 RuntimeTest[3189:156810] instance variable's name: _string at index: 3

2014-10-22 19:41:37.463 RuntimeTest[3189:156810] instance variable's name: _integer at index: 4

2014-10-22 19:41:37.463 RuntimeTest[3189:156810] instace variable _string

2014-10-22 19:41:37.463 RuntimeTest[3189:156810] ==========================================================

2014-10-22 19:41:37.463 RuntimeTest[3189:156810] property's name: array

2014-10-22 19:41:37.463 RuntimeTest[3189:156810] property's name: string

2014-10-22 19:41:37.464 RuntimeTest[3189:156810] property's name: integer

2014-10-22 19:41:37.464 RuntimeTest[3189:156810] property array

2014-10-22 19:41:37.464 RuntimeTest[3189:156810] ==========================================================

2014-10-22 19:41:37.464 RuntimeTest[3189:156810] method's signature: method1

2014-10-22 19:41:37.464 RuntimeTest[3189:156810] method's signature: method2

2014-10-22 19:41:37.464 RuntimeTest[3189:156810] method's signature: method3WithArg1:arg2:

2014-10-22 19:41:37.465 RuntimeTest[3189:156810] method's signature: integer

2014-10-22 19:41:37.465 RuntimeTest[3189:156810] method's signature: setInteger:

2014-10-22 19:41:37.465 RuntimeTest[3189:156810] method's signature: array

2014-10-22 19:41:37.465 RuntimeTest[3189:156810] method's signature: string

2014-10-22 19:41:37.465 RuntimeTest[3189:156810] method's signature: setString:

2014-10-22 19:41:37.465 RuntimeTest[3189:156810] method's signature: setArray:

2014-10-22 19:41:37.466 RuntimeTest[3189:156810] method's signature: .cxx_destruct

2014-10-22 19:41:37.466 RuntimeTest[3189:156810] method method1

2014-10-22 19:41:37.466 RuntimeTest[3189:156810] class method : classMethod1

2014-10-22 19:41:37.466 RuntimeTest[3189:156810] MyClass is responsd to selector: method3WithArg1:arg2:

2014-10-22 19:41:37.467 RuntimeTest[3189:156810] call method method1

2014-10-22 19:41:37.467 RuntimeTest[3189:156810] ==========================================================

2014-10-22 19:41:37.467 RuntimeTest[3189:156810] protocol name: NSCopying

2014-10-22 19:41:37.467 RuntimeTest[3189:156810] protocol name: NSCoding

2014-10-22 19:41:37.467 RuntimeTest[3189:156810] MyClass is responsed to protocol NSCoding

2014-10-22 19:41:37.468 RuntimeTest[3189:156810] ==========================================================

動態創建類和對象

runtime的強大之處在于它能在運行時創建類和對象。

動態創建類

動態創建類涉及到以下幾個函數:

// 創建一個新類和元類

Class objc_allocateClassPair ( Class superclass,constchar*name, size_t extraBytes );

// 銷毀一個類及其相關聯的類

voidobjc_disposeClassPair ( Class cls );

// 在應用中注冊由objc_allocateClassPair創建的類

voidobjc_registerClassPair ( Class cls );

objc_allocateClassPair函數:如果我們要創建一個根類,則superclass指定為Nil。extraBytes通常指定為0,該參數是分配給類和元類對象尾部的索引ivars的字節數。

為了創建一個新類,我們需要調用objc_allocateClassPair。然后使用諸如class_addMethod,class_addIvar等函數來為新創建的類添加方法、實例變量和屬性等。完成這些后,我們需要調用objc_registerClassPair函數來注冊類,之后這個新類就可以在程序中使用了。

實例方法和實例變量應該添加到類自身上,而類方法應該添加到類的元類上。

objc_disposeClassPair函數用于銷毀一個類,不過需要注意的是,如果程序運行中還存在類或其子類的實例,則不能調用針對類調用該方法。

在前面介紹元類時,我們已經有接觸到這幾個函數了,在此我們再舉個實例來看看這幾個函數的使用。

Class cls = objc_allocateClassPair(MyClass.class,"MySubClass",0);

class_addMethod(cls,@selector(submethod1), (IMP)imp_submethod1,"v@:");

class_replaceMethod(cls,@selector(method1), (IMP)imp_submethod1,"v@:");

class_addIvar(cls,"_ivar1",sizeof(NSString*), log(sizeof(NSString*)),"i");

objc_property_attribute_t type = {"T","@\"NSString\""};

objc_property_attribute_t ownership = {"C",""};

objc_property_attribute_t backingivar = {"V","_ivar1"};

objc_property_attribute_t attrs[] = {type, ownership, backingivar};

class_addProperty(cls,"property2", attrs,3);

objc_registerClassPair(cls);

idinstance = [[cls alloc] init];

[instance performSelector:@selector(submethod1)];

[instance performSelector:@selector(method1)];

程序的輸出如下:

2014-10-2311:35:31.006RuntimeTest[3800:66152] run sub method1

2014-10-2311:35:31.006RuntimeTest[3800:66152] run sub method1

動態創建對象

動態創建對象的函數如下:

// 創建類實例

idclass_createInstance ( Class cls, size_t extraBytes );

// 在指定位置創建類實例

idobjc_constructInstance ( Class cls,void*bytes );

// 銷毀類實例

void* objc_destructInstance (idobj );

class_createInstance函數:創建實例時,會在默認的內存區域為類分配內存。extraBytes參數表示分配的額外字節數。這些額外的字節可用于存儲在類定義中所定義的實例變量之外的實例變量。該函數在ARC環境下無法使用。

調用class_createInstance的效果與+alloc方法類似。不過在使用class_createInstance時,我們需要確切的知道我們要用它來做什么。在下面的例子中,我們用NSString來測試一下該函數的實際效果:

id theObject = class_createInstance(NSString.class,sizeof(unsigned));

id str1 = [theObject init];

NSLog(@"%@", [str1class]);

idstr2 = [[NSStringalloc] initWithString:@"test"];

NSLog(@"%@", [str2class]);

輸出結果是:

2014-10-2312:46:50.781RuntimeTest[4039:89088]NSString

2014-10-2312:46:50.781RuntimeTest[4039:89088] __NSCFConstantString

可以看到,使用class_createInstance函數獲取的是NSString實例,而不是類簇中的默認占位符類__NSCFConstantString。

objc_constructInstance函數:在指定的位置(bytes)創建類實例。

objc_destructInstance函數:銷毀一個類的實例,但不會釋放并移除任何與其相關的引用。

實例操作函數

實例操作函數主要是針對我們創建的實例對象的一系列操作函數,我們可以使用這組函數來從實例對象中獲取我們想要的一些信息,如實例對象中變量的值。這組函數可以分為三小類:

1.針對整個對象進行操作的函數,這類函數包含

// 返回指定對象的一份拷貝

idobject_copy (idobj, size_t size );

// 釋放指定對象占用的內存

idobject_dispose (idobj );

有這樣一種場景,假設我們有類A和類B,且類B是類A的子類。類B通過添加一些額外的屬性來擴展類A。現在我們創建了一個A類的實例對象,并希望在運行時將這個對象轉換為B類的實例對象,這樣可以添加數據到B類的屬性中。這種情況下,我們沒有辦法直接轉換,因為B類的實例會比A類的實例更大,沒有足夠的空間來放置對象。此時,我們就要以使用以上幾個函數來處理這種情況,如下代碼所示:

NSObject*a = [[NSObjectalloc] init];

idnewB = object_copy(a, class_getInstanceSize(MyClass.class));

object_setClass(newB, MyClass.class);

object_dispose(a);

2.針對對象實例變量進行操作的函數,這類函數包含:

// 修改類實例的實例變量的值

Ivar object_setInstanceVariable (idobj,constchar*name,void*value );

// 獲取對象實例變量的值

Ivar object_getInstanceVariable (idobj,constchar*name,void**outValue );

// 返回指向給定對象分配的任何額外字節的指針

void* object_getIndexedIvars (idobj );

// 返回對象中實例變量的值

idobject_getIvar (idobj, Ivar ivar );

// 設置對象中實例變量的值

voidobject_setIvar (idobj, Ivar ivar,idvalue );

如果實例變量的Ivar已經知道,那么調用object_getIvar會比object_getInstanceVariable函數快,相同情況下,object_setIvar也比object_setInstanceVariable快。

3.針對對象的類進行操作的函數,這類函數包含:

// 返回給定對象的類名

constchar* object_getClassName (idobj );

// 返回對象的類

Class object_getClass (idobj );

// 設置對象的類

Class object_setClass (idobj, Class cls );

獲取類定義

Objective-C動態運行庫會自動注冊我們代碼中定義的所有的類。我們也可以在運行時創建類定義并使用objc_addClass函數來注冊它們。runtime提供了一系列函數來獲取類定義相關的信息,這些函數主要包括:

// 獲取已注冊的類定義的列表

intobjc_getClassList ( Class *buffer,intbufferCount );

// 創建并返回一個指向所有已注冊類的指針列表

Class * objc_copyClassList (unsignedint*outCount );

// 返回指定類的類定義

Class objc_lookUpClass (constchar*name );

Class objc_getClass (constchar*name );

Class objc_getRequiredClass (constchar*name );

// 返回指定類的元類

Class objc_getMetaClass (constchar*name );

objc_getClassList函數:獲取已注冊的類定義的列表。我們不能假設從該函數中獲取的類對象是繼承自NSObject體系的,所以在這些類上調用方法是,都應該先檢測一下這個方法是否在這個類中實現。

下面代碼演示了該函數的用法:

int numClasses;

Class * classes =NULL;

numClasses = objc_getClassList(NULL,0);

if(numClasses >0) {

classes = malloc(sizeof(Class) * numClasses);

numClasses = objc_getClassList(classes, numClasses);

NSLog(@"number of classes: %d", numClasses);

for(inti =0; i < numClasses; i++) {

Class cls = classes[i];

NSLog(@"class name: %s", class_getName(cls));

}

free(classes);

}

輸出結果如下:

2014-10-2316:20:52.589RuntimeTest[8437:188589] number of classes:1282

2014-10-2316:20:52.589RuntimeTest[8437:188589]classname: DDTokenRegexp

2014-10-2316:20:52.590RuntimeTest[8437:188589]classname: _NSMostCommonKoreanCharsKeySet

2014-10-2316:20:52.590RuntimeTest[8437:188589]classname: OS_xpc_dictionary

2014-10-2316:20:52.590RuntimeTest[8437:188589]classname:NSFileCoordinator

2014-10-2316:20:52.590RuntimeTest[8437:188589]classname:NSAssertionHandler

2014-10-2316:20:52.590RuntimeTest[8437:188589]classname: PFUbiquityTransactionLogMigrator

2014-10-2316:20:52.591RuntimeTest[8437:188589]classname:NSNotification

2014-10-2316:20:52.591RuntimeTest[8437:188589]classname:NSKeyValueNilSetEnumerator

2014-10-2316:20:52.591RuntimeTest[8437:188589]classname: OS_tcp_connection_tls_session

2014-10-2316:20:52.591RuntimeTest[8437:188589]classname: _PFRoutines

......還有大量輸出

獲取類定義的方法有三個:objc_lookUpClass,objc_getClass和objc_getRequiredClass。如果類在運行時未注冊,則objc_lookUpClass會返回nil,而objc_getClass會調用類處理回調,并再次確認類是否注冊,如果確認未注冊,再返回nil。而objc_getRequiredClass函數的操作與objc_getClass相同,只不過如果沒有找到類,則會殺死進程。

objc_getMetaClass函數:如果指定的類沒有注冊,則該函數會調用類處理回調,并再次確認類是否注冊,如果確認未注冊,再返回nil。不過,每個類定義都必須有一個有效的元類定義,所以這個函數總是會返回一個元類定義,不管它是否有效。

小結

在這一章中我們介紹了Runtime運行時中與類和對象相關的數據結構,通過這些數據函數,我們可以管窺Objective-C底層面向對象實現的一些信息。另外,通過豐富的操作函數,可以靈活地對這些數據進行操作。

最后編輯于
?著作權歸作者所有,轉載或內容合作請聯系作者
平臺聲明:文章內容(如有圖片或視頻亦包括在內)由作者上傳并發布,文章內容僅代表作者本人觀點,簡書系信息發布平臺,僅提供信息存儲服務。
  • 序言:七十年代末,一起剝皮案震驚了整個濱河市,隨后出現的幾起案子,更是在濱河造成了極大的恐慌,老刑警劉巖,帶你破解...
    沈念sama閱讀 227,797評論 6 531
  • 序言:濱河連續發生了三起死亡事件,死亡現場離奇詭異,居然都是意外死亡,警方通過查閱死者的電腦和手機,發現死者居然都...
    沈念sama閱讀 98,179評論 3 414
  • 文/潘曉璐 我一進店門,熙熙樓的掌柜王于貴愁眉苦臉地迎上來,“玉大人,你說我怎么就攤上這事。” “怎么了?”我有些...
    開封第一講書人閱讀 175,628評論 0 373
  • 文/不壞的土叔 我叫張陵,是天一觀的道長。 經常有香客問我,道長,這世上最難降的妖魔是什么? 我笑而不...
    開封第一講書人閱讀 62,642評論 1 309
  • 正文 為了忘掉前任,我火速辦了婚禮,結果婚禮上,老公的妹妹穿的比我還像新娘。我一直安慰自己,他們只是感情好,可當我...
    茶點故事閱讀 71,444評論 6 405
  • 文/花漫 我一把揭開白布。 她就那樣靜靜地躺著,像睡著了一般。 火紅的嫁衣襯著肌膚如雪。 梳的紋絲不亂的頭發上,一...
    開封第一講書人閱讀 54,948評論 1 321
  • 那天,我揣著相機與錄音,去河邊找鬼。 笑死,一個胖子當著我的面吹牛,可吹牛的內容都是我干的。 我是一名探鬼主播,決...
    沈念sama閱讀 43,040評論 3 440
  • 文/蒼蘭香墨 我猛地睜開眼,長吁一口氣:“原來是場噩夢啊……” “哼!你這毒婦竟也來了?” 一聲冷哼從身側響起,我...
    開封第一講書人閱讀 42,185評論 0 287
  • 序言:老撾萬榮一對情侶失蹤,失蹤者是張志新(化名)和其女友劉穎,沒想到半個月后,有當地人在樹林里發現了一具尸體,經...
    沈念sama閱讀 48,717評論 1 333
  • 正文 獨居荒郊野嶺守林人離奇死亡,尸身上長有42處帶血的膿包…… 初始之章·張勛 以下內容為張勛視角 年9月15日...
    茶點故事閱讀 40,602評論 3 354
  • 正文 我和宋清朗相戀三年,在試婚紗的時候發現自己被綠了。 大學時的朋友給我發了我未婚夫和他白月光在一起吃飯的照片。...
    茶點故事閱讀 42,794評論 1 369
  • 序言:一個原本活蹦亂跳的男人離奇死亡,死狀恐怖,靈堂內的尸體忽然破棺而出,到底是詐尸還是另有隱情,我是刑警寧澤,帶...
    沈念sama閱讀 38,316評論 5 358
  • 正文 年R本政府宣布,位于F島的核電站,受9級特大地震影響,放射性物質發生泄漏。R本人自食惡果不足惜,卻給世界環境...
    茶點故事閱讀 44,045評論 3 347
  • 文/蒙蒙 一、第九天 我趴在偏房一處隱蔽的房頂上張望。 院中可真熱鬧,春花似錦、人聲如沸。這莊子的主人今日做“春日...
    開封第一講書人閱讀 34,418評論 0 26
  • 文/蒼蘭香墨 我抬頭看了看天上的太陽。三九已至,卻和暖如春,著一層夾襖步出監牢的瞬間,已是汗流浹背。 一陣腳步聲響...
    開封第一講書人閱讀 35,671評論 1 281
  • 我被黑心中介騙來泰國打工, 沒想到剛下飛機就差點兒被人妖公主榨干…… 1. 我叫王不留,地道東北人。 一個月前我還...
    沈念sama閱讀 51,414評論 3 390
  • 正文 我出身青樓,卻偏偏與公主長得像,于是被迫代替她去往敵國和親。 傳聞我的和親對象是個殘疾皇子,可洞房花燭夜當晚...
    茶點故事閱讀 47,750評論 2 370

推薦閱讀更多精彩內容