Objective-C語(yǔ)言是一門動(dòng)態(tài)語(yǔ)言,它將很多靜態(tài)語(yǔ)言在編譯和鏈接時(shí)期做的事放到了運(yùn)行時(shí)來(lái)處理。這種動(dòng)態(tài)語(yǔ)言的優(yōu)勢(shì)在于:我們寫代碼時(shí)更具靈活性,如我們可以把消息轉(zhuǎn)發(fā)給我們想要的對(duì)象,或者隨意交換一個(gè)方法的實(shí)現(xiàn)等。
這種特性意味著Objective-C不僅需要一個(gè)編譯器,還需要一個(gè)運(yùn)行時(shí)系統(tǒng)來(lái)執(zhí)行編譯的代碼。對(duì)于Objective-C來(lái)說(shuō),這個(gè)運(yùn)行時(shí)系統(tǒng)就像一個(gè)操作系統(tǒng)一樣:它讓所有的工作可以正常的運(yùn)行。這個(gè)運(yùn)行時(shí)系統(tǒng)即Objc Runtime
。Objc Runtime
其實(shí)是一個(gè)Runtime
庫(kù),它基本上是用C和匯編寫的,這個(gè)庫(kù)使得C語(yǔ)言有了面向?qū)ο蟮哪芰Α?br>
<h5>Runtime<h5>
庫(kù)主要做下面幾件事:
封裝:在這個(gè)庫(kù)中,對(duì)象可以用C語(yǔ)言中的結(jié)構(gòu)體表示,而方法可以用C函數(shù)來(lái)實(shí)現(xiàn),另外再加上了一些額外的特性。這些結(jié)構(gòu)體和函數(shù)被runtime函數(shù)封裝后,我們就可以在程序運(yùn)行時(shí)創(chuàng)建,檢查,修改類、對(duì)象和它們的方法了。
找出方法的最終執(zhí)行代碼:當(dāng)程序執(zhí)行[object doSomething]
時(shí),會(huì)向消息接收者(object)發(fā)送一條消息(doSomething),runtime會(huì)根據(jù)消息接收者是否能響應(yīng)該消息而做出不同的反應(yīng)。這將在后面詳細(xì)介紹。
Objective-C runtime目前有兩個(gè)版本:Modern runtime
和Legacy runtime
。Modern Runtime
覆蓋了64位的Mac OS X Apps
,還有iOS Apps
,Legacy Runtime
是早期用來(lái)給32位 Mac OS X Apps
用的,也就是可以不用管就是了。
在這一系列文章中,我們將介紹runtime的基本工作原理,以及如何利用它讓我們的程序變得更加靈活。在本文中,我們先來(lái)介紹一下類與對(duì)象,這是面向?qū)ο蟮幕A(chǔ),我們看看在Runtime中,類是如何實(shí)現(xiàn)的。
類與對(duì)象基礎(chǔ)數(shù)據(jù)結(jié)構(gòu)
Class
Objective-C類是由Class
類型來(lái)表示的,它實(shí)際上是一個(gè)指向objc_class
結(jié)構(gòu)體的指針。它的定義如下:
<code>
typedef struct objc_class *Class;
</code>
查看objc/runtime.h
中objc_class
結(jié)構(gòu)體的定義如下:
<code>
struct objc_class {
Class isa OBJC_ISA_AVAILABILITY;
if !OBJC2
Class super_class OBJC2_UNAVAILABLE; // 父類
const char *name OBJC2_UNAVAILABLE; // 類名
long version OBJC2_UNAVAILABLE; // 類的版本信息,默認(rèn)為0
long info OBJC2_UNAVAILABLE; // 類信息,供運(yùn)行期使用的一些位標(biāo)識(shí)
long instance_size OBJC2_UNAVAILABLE; // 該類的實(shí)例變量大小
struct objc_ivar_list *ivars OBJC2_UNAVAILABLE; // 該類的成員變量鏈表
struct objc_method_list **methodLists OBJC2_UNAVAILABLE; // 方法定義的鏈表
struct objc_cache *cache OBJC2_UNAVAILABLE; // 方法緩存
struct objc_protocol_list *protocols OBJC2_UNAVAILABLE; // 協(xié)議鏈表
endif
} OBJC2_UNAVAILABLE;
</code>
在這個(gè)定義中,下面幾個(gè)字段是我們感興趣的
isa
:需要注意的是在Objective-C中,所有的類自身也是一個(gè)對(duì)象,這個(gè)對(duì)象的Class
里面也有一個(gè)isa
指針,它指向metaClass
(元類),我們會(huì)在后面介紹它。
super_class
:指向該類的父類,如果該類已經(jīng)是最頂層的根類(如NSObject或NSProxy),則super_class
為NULL。
cache
:用于緩存最近使用的方法。一個(gè)接收者對(duì)象接收到一個(gè)消息時(shí),它會(huì)根據(jù)isa
指針去查找能夠響應(yīng)這個(gè)消息的對(duì)象。在實(shí)際使用中,這個(gè)對(duì)象只有一部分方法是常用的,很多方法其實(shí)很少用或者根本用不上。這種情況下,如果每次消息來(lái)時(shí),我們都是methodLists
中遍歷一遍,性能勢(shì)必很差。這時(shí),cache
就派上用場(chǎng)了。在我們每次調(diào)用過(guò)一個(gè)方法后,這個(gè)方法就會(huì)被緩存到cache
列表中,下次調(diào)用的時(shí)候runtime就會(huì)優(yōu)先去cache
中查找,如果cache
沒(méi)有,才去methodLists
中查找方法。這樣,對(duì)于那些經(jīng)常用到的方法的調(diào)用,但提高了調(diào)用的效率。
version
:我們可以使用這個(gè)字段來(lái)提供類的版本信息。這對(duì)于對(duì)象的序列化非常有用,它可是讓我們識(shí)別出不同類定義版本中實(shí)例變量布局的改變。
針對(duì)cache
,我們用下面例子來(lái)說(shuō)明其執(zhí)行過(guò)程:
<code>
NSArray *array = [[NSArray alloc] init];
</code>
其流程是:
1. `[NSArray alloc]`先被執(zhí)行。因?yàn)镹SArray沒(méi)有`+alloc`方法,于是去父類NSObject去查找。
2. 檢測(cè)NSObject是否響應(yīng)`+alloc`方法,發(fā)現(xiàn)響應(yīng),于是檢測(cè)NSArray類,并根據(jù)其所需的內(nèi)存空間大小開始分配內(nèi)存空間,然后把`isa`指針指向NSArray類。同時(shí),`+alloc`也被加進(jìn)cache列表里面。
3. 接著,執(zhí)行`-init`方法,如果NSArray響應(yīng)該方法,則直接將其加入`cache`;如果不響應(yīng),則去父類查找。
4. 在后期的操作中,如果再以`[[NSArray alloc] init]`這種方式來(lái)創(chuàng)建數(shù)組,則會(huì)直接從cache中取出相應(yīng)的方法,直接調(diào)用。
### objc_object與id
`objc_object`是表示一個(gè)類的實(shí)例的結(jié)構(gòu)體,它的定義如下(`objc/objc.h`):
<code>
```objc
struct objc_object {
Class isa OBJC_ISA_AVAILABILITY;
};
typedef struct objc_object *id;
</code>
可以看到,這個(gè)結(jié)構(gòu)體只有一個(gè)字體,即指向其類的isa
指針。這樣,當(dāng)我們向一個(gè)Objective-C對(duì)象發(fā)送消息時(shí),運(yùn)行時(shí)庫(kù)會(huì)根據(jù)實(shí)例對(duì)象的isa
指針找到這個(gè)實(shí)例對(duì)象所屬的類。Runtime
庫(kù)會(huì)在類的方法列表及父類的方法列表中去尋找與消息對(duì)應(yīng)的selector
指向的方法。找到后即運(yùn)行這個(gè)方法。
當(dāng)創(chuàng)建一個(gè)特定類的實(shí)例對(duì)象時(shí),分配的內(nèi)存包含一個(gè)objc_object
數(shù)據(jù)結(jié)構(gòu),然后是類的實(shí)例變量的數(shù)據(jù)。NSObject類的alloc
和allocWithZone:
方法使用函數(shù)class_createInstance
來(lái)創(chuàng)建objc_object
數(shù)據(jù)結(jié)構(gòu)。
另外還有我們常見的id,它是一個(gè)objc_object
結(jié)構(gòu)類型的指針。它的存在可以讓我們實(shí)現(xiàn)類似于C++中泛型的一些操作。該類型的對(duì)象可以轉(zhuǎn)換為任何一種對(duì)象,有點(diǎn)類似于C語(yǔ)言中void *
指針類型的作用。
[](http://southpeak.github.io/2014/10/25/objective-c-runtime-1/#objc-cache)objc_cache
上面提到了objc_class
結(jié)構(gòu)體中的cache
字段,它用于緩存調(diào)用過(guò)的方法。這個(gè)字段是一個(gè)指向objc_cache
結(jié)構(gòu)體的指針,其定義如下:
<code>
struct objc_cache {
unsigned int mask /* total = mask + 1 */ OBJC2_UNAVAILABLE;
unsigned int occupied OBJC2_UNAVAILABLE;
Method buckets[1] OBJC2_UNAVAILABLE;
};
</code>
該結(jié)構(gòu)體的字段描述如下:
mask
:一個(gè)整數(shù),指定分配的緩存bucket
的總數(shù)。在方法查找過(guò)程中,Objective-C runtime使用這個(gè)字段來(lái)確定開始線性查找數(shù)組的索引位置。指向方法selector
的指針與該字段做一個(gè)AND
位操作(index = (mask & selector))
。這可以作為一個(gè)簡(jiǎn)單的hash
散列算法。
occupied
:一個(gè)整數(shù),指定實(shí)際占用的緩存bucket
的總數(shù)。
buckets
:指向Method
數(shù)據(jù)結(jié)構(gòu)指針的數(shù)組。這個(gè)數(shù)組可能包含不超過(guò)mask+1
個(gè)元素。需要注意的是,指針可能是NULL,表示這個(gè)緩存bucket
沒(méi)有被占用,另外被占用的bucket
可能是不連續(xù)的。這個(gè)數(shù)組可能會(huì)隨著時(shí)間而增長(zhǎng)。
[](http://southpeak.github.io/2014/10/25/objective-c-runtime-1/#元類-Meta-Class)元類(Meta Class)
在上面我們提到,所有的類自身也是一個(gè)對(duì)象,我們可以向這個(gè)對(duì)象發(fā)送消息(即調(diào)用類方法)。如:
1
NSArray *array = [NSArray array];
這個(gè)例子中,+array
消息發(fā)送給了NSArray類,而這個(gè)NSArray也是一個(gè)對(duì)象。既然是對(duì)象,那么它也是一個(gè)objc_object
指針,它包含一個(gè)指向其類的一個(gè)isa
指針。那么這些就有一個(gè)問(wèn)題了,這個(gè)isa
指針指向什么呢?為了調(diào)用+array
方法,這個(gè)類的isa
指針必須指向一個(gè)包含這些類方法的一個(gè)objc_class
結(jié)構(gòu)體。這就引出了meta-class
的概念
meta-class是一個(gè)類對(duì)象的類。
當(dāng)我們向一個(gè)對(duì)象發(fā)送消息時(shí),runtime會(huì)在這個(gè)對(duì)象所屬的這個(gè)類的方法列表中查找方法;而向一個(gè)類發(fā)送消息時(shí),會(huì)在這個(gè)類的meta-class
的方法列表中查找。
meta-class
之所以重要,是因?yàn)樗鎯?chǔ)著一個(gè)類的所有類方法。每個(gè)類都會(huì)有一個(gè)單獨(dú)的meta-class
,因?yàn)槊總€(gè)類的類方法基本不可能完全相同。
再深入一下,meta-class
也是一個(gè)類,也可以向它發(fā)送一個(gè)消息,那么它的isa
又是指向什么呢?為了不讓這種結(jié)構(gòu)無(wú)限延伸下去,Objective-C的設(shè)計(jì)者讓所有的meta-class
的isa
指向基類的meta-class
,以此作為它們的所屬類。即,任何NSObject繼承體系下的meta-class
都使用NSObject的meta-class
作為自己的所屬類,而基類的meta-class
的isa
指針是指向它自己。這樣就形成了一個(gè)完美的閉環(huán)。
通過(guò)上面的描述,再加上對(duì)objc_class
結(jié)構(gòu)體中super_class
指針的分析,我們就可以描繪出類及相應(yīng)meta-class
對(duì)于NSObject繼承體系來(lái)說(shuō),其實(shí)例方法對(duì)體系中的所有實(shí)例、類和meta-class
都是有效的;而類方法對(duì)于體系內(nèi)的所有類和meta-class
都是有效的。
講了這么多,我們還是來(lái)寫個(gè)例子吧:
<code>
void TestMetaClass(id self, SEL _cmd) {
NSLog(@"This objcet is %p", self);
NSLog(@"Class is %@, super class is %@", [self class], [self superclass]);
Class currentClass = [self class];
for (int i = 0; i < 4; i++) {
NSLog(@"Following the isa pointer %d times gives %p", i, currentClass);
currentClass = objc_getClass((__bridge void *)currentClass);
}
NSLog(@"NSObject's class is %p", [NSObject class]);
NSLog(@"NSObject's meta class is %p", objc_getClass((__bridge void *)[NSObject class]));
}
#pragma mark -
@implementation Test
- (void)ex_registerClassPair {
Class newClass = objc_allocateClassPair([NSError class], "TestClass", 0);
class_addMethod(newClass, @selector(testMetaClass), (IMP)TestMetaClass, "v@:");
objc_registerClassPair(newClass);
id instance = [[newClass alloc] initWithDomain:@"some domain" code:0 userInfo:nil];
[instance performSelector:@selector(testMetaClass)];
}
@end
</code>
這個(gè)例子是在運(yùn)行時(shí)創(chuàng)建了一個(gè)NSError
的子類TestClass
,然后為這個(gè)子類添加一個(gè)方法testMetaClass
,這個(gè)方法的實(shí)現(xiàn)是TestMetaClass
函數(shù)。
運(yùn)行后,打印結(jié)果是
<code>
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
</code>
我們?cè)趂or循環(huán)中,我們通過(guò)objc_getClass
來(lái)獲取對(duì)象的isa
,并將其打印出來(lái),依此一直回溯到NSObject的meta-class
。分析打印結(jié)果,可以看到最后指針指向的地址是0x0,即NSObject的meta-class
的類地址。
*這里需要注意的是:我們?cè)谝粋€(gè)類對(duì)象調(diào)用class方法是無(wú)法獲取meta-class,它只是返回類而已。*
[](http://southpeak.github.io/2014/10/25/objective-c-runtime-1/#類與對(duì)象操作函數(shù))類與對(duì)象操作函數(shù)
runtime提供了大量的函數(shù)來(lái)操作類與對(duì)象。類的操作方法大部分是以class_
為前綴的,而對(duì)象的操作方法大部分是以objc_
或object_
為前綴。下面我們將根據(jù)這些方法的用途來(lái)分類討論這些方法的使用。
[](http://southpeak.github.io/2014/10/25/objective-c-runtime-1/#類相關(guān)操作函數(shù))類相關(guān)操作函數(shù)
我們可以回過(guò)頭去看看objc_class
的定義,runtime提供的操作類的方法主要就是針對(duì)這個(gè)結(jié)構(gòu)體中的各個(gè)字段的。下面我們分別介紹這一些的函數(shù)。并在最后以實(shí)例來(lái)演示這些函數(shù)的具體用法。
[](http://southpeak.github.io/2014/10/25/objective-c-runtime-1/#類名-name)類名(name)
類名操作的函數(shù)主要有:
// 獲取類的類名
const char * class_getName ( Class cls );
對(duì)于class_getName
函數(shù),如果傳入的cls
為Nil
,則返回一個(gè)字字符串。
[](http://southpeak.github.io/2014/10/25/objective-c-runtime-1/#父類-super-class-和元類-meta-class)父類(super_class)和元類(meta-class)
父類和元類操作的函數(shù)主要有:
// 獲取類的父類
Class class_getSuperclass ( Class cls );
// 判斷給定的Class是否是一個(gè)元類
BOOL class_isMetaClass ( Class cls );
class_getSuperclass
函數(shù),當(dāng)cls
為Nil或者cls
為根類時(shí),返回Nil。不過(guò)通常我們可以使用NSObject類的superclass
方法來(lái)達(dá)到同樣的目的。
class_isMetaClass
函數(shù),如果是cls
是元類,則返回YES;如果否或者傳入的cls
為Nil,則返回NO。
[](http://southpeak.github.io/2014/10/25/objective-c-runtime-1/#實(shí)例變量大小-instance-size)實(shí)例變量大小(instance_size)
實(shí)例變量大小操作的函數(shù)有:
// 獲取實(shí)例大小
size_t class_getInstanceSize ( Class cls );
[](http://southpeak.github.io/2014/10/25/objective-c-runtime-1/#成員變量-ivars-及屬性)成員變量(ivars)及屬性
在objc_class
中,所有的成員變量、屬性的信息是放在鏈表ivars
中的。ivars
是一個(gè)數(shù)組,數(shù)組中每個(gè)元素是指向Ivar
(變量信息)的指針。runtime提供了豐富的函數(shù)來(lái)操作這一字段。大體上可以分為以下幾類:
1.成員變量操作函數(shù),主要包含以下函數(shù):
<code>
// 獲取類中指定名稱實(shí)例成員變量的信息
Ivar class_getInstanceVariable ( Class cls, const char *name );
// 獲取類成員變量的信息
Ivar class_getClassVariable ( Class cls, const char *name );
// 添加成員變量
BOOL class_addIvar ( Class cls, const char *name, size_t size, uint8_t alignment, const char *types );
// 獲取整個(gè)成員變量列表
Ivar * class_copyIvarList ( Class cls, unsigned int *outCount );
class_getInstanceVariable
函數(shù),它返回一個(gè)指向包含name指定的成員變量信息的objc_ivar
結(jié)構(gòu)體的指針(Ivar
)。
</code>
**class_getClassVariable**
函數(shù),目前沒(méi)有找到關(guān)于Objective-C中類變量的信息,一般認(rèn)為Objective-C不支持類變量。注意,返回的列表不包含父類的成員變量和屬性。
Objective-C不支持往已存在的類中添加實(shí)例變量,因此不管是系統(tǒng)庫(kù)提供的提供的類,還是我們自定義的類,都無(wú)法動(dòng)態(tài)添加成員變量。但如果我們通過(guò)運(yùn)行時(shí)來(lái)創(chuàng)建一個(gè)類的話,又應(yīng)該如何給它添加成員變量呢?這時(shí)我們就可以使用class_addIvar
函數(shù)了。不過(guò)需要注意的是,這個(gè)方法只能在objc_allocateClassPair
函數(shù)與objc_registerClassPair
之間調(diào)用。另外,這個(gè)類也不能是元類。成員變量的按字節(jié)最小對(duì)齊量是1<<alignment
。這取決于ivar
的類型和機(jī)器的架構(gòu)。如果變量的類型是指針類型,則傳遞log2(sizeof(pointer_type))
。
**class_copyIvarList**
函數(shù),它返回一個(gè)指向成員變量信息的數(shù)組,數(shù)組中每個(gè)元素是指向該成員變量信息的objc_ivar
結(jié)構(gòu)體的指針。這個(gè)數(shù)組不包含在父類中聲明的變量。outCount
指針?lè)祷財(cái)?shù)組的大小。需要注意的是,我們必須使用free()
來(lái)釋放這個(gè)數(shù)組。
2.屬性操作函數(shù),主要包含以下函數(shù):
<code>
// 獲取指定的屬性
objc_property_t class_getProperty ( Class cls, const char *name );
// 獲取屬性列表
objc_property_t * class_copyPropertyList ( Class cls, unsigned int *outCount );
// 為類添加屬性
BOOL class_addProperty ( Class cls, const char *name, const objc_property_attribute_t *attributes, unsigned int attributeCount );
// 替換類的屬性
void class_replaceProperty ( Class cls, const char *name, const objc_property_attribute_t *attributes, unsigned int attributeCount );
</code>
這一種方法也是針對(duì)ivars
來(lái)操作,不過(guò)只操作那些是屬性的值。我們?cè)诤竺娼榻B屬性時(shí)會(huì)再遇到這些函數(shù)。
3.在MAC OS X系統(tǒng)中,我們可以使用垃圾回收器。runtime提供了幾個(gè)函數(shù)來(lái)確定一個(gè)對(duì)象的內(nèi)存區(qū)域是否可以被垃圾回收器掃描,以處理strong/weak
引用。這幾個(gè)函數(shù)定義如下:
<code>
const uint8_t * class_getIvarLayout ( Class cls );
void class_setIvarLayout ( Class cls, const uint8_t *layout );
const uint8_t * class_getWeakIvarLayout ( Class cls );
void class_setWeakIvarLayout ( Class cls, const uint8_t *layout );
</code>
但通常情況下,我們不需要去主動(dòng)調(diào)用這些方法;在調(diào)用objc_registerClassPair
時(shí),會(huì)生成合理的布局。在此不詳細(xì)介紹這些函數(shù)。
[](http://southpeak.github.io/2014/10/25/objective-c-runtime-1/#方法-methodLists)方法(methodLists)
方法操作主要有以下函數(shù):
<code>
// 添加方法
BOOL class_addMethod ( Class cls, SEL name, IMP imp, const char *types );
// 獲取實(shí)例方法
Method class_getInstanceMethod ( Class cls, SEL name );
// 獲取類方法
Method class_getClassMethod ( Class cls, SEL name );
// 獲取所有方法的數(shù)組
Method * class_copyMethodList ( Class cls, unsigned int *outCount );
// 替代方法的實(shí)現(xiàn)
IMP class_replaceMethod ( Class cls, SEL name, IMP imp, const char *types );
// 返回方法的具體實(shí)現(xiàn)
IMP class_getMethodImplementation ( Class cls, SEL name );
IMP class_getMethodImplementation_stret ( Class cls, SEL name );
// 類實(shí)例是否響應(yīng)指定的selector
BOOL class_respondsToSelector ( Class cls, SEL sel );
</code>
**class_addMethod**
的實(shí)現(xiàn)會(huì)覆蓋父類的方法實(shí)現(xiàn),但不會(huì)取代本類中已存在的實(shí)現(xiàn),如果本類中包含一個(gè)同名的實(shí)現(xiàn),則函數(shù)會(huì)返回NO。如果要修改已存在實(shí)現(xiàn),可以使用method_setImplementation
。一個(gè)Objective-C方法是一個(gè)簡(jiǎn)單的C函數(shù),它至少包含兩個(gè)參數(shù)–self
和_cmd
。所以,我們的實(shí)現(xiàn)函數(shù)(IMP參數(shù)指向的函數(shù))至少需要兩個(gè)參數(shù),如下所示:
<code>
void myMethodIMP(id self, SEL _cmd)
{
// implementation ....
}
</code>
與成員變量不同的是,我們可以為類動(dòng)態(tài)添加方法,不管這個(gè)類是否已存在。
另外,參數(shù)types
是一個(gè)描述傳遞給方法的參數(shù)類型的字符數(shù)組,這就涉及到類型編碼,我們將在后面介紹。
class_getInstanceMethod
、class_getClassMethod
函數(shù),與class_copyMethodList
不同的是,這兩個(gè)函數(shù)都會(huì)去搜索父類的實(shí)現(xiàn)。
**class_copyMethodList**
函數(shù),返回包含所有實(shí)例方法的數(shù)組,如果需要獲取類方法,則可以使用class_copyMethodList(object_getClass(cls), &count)
(一個(gè)類的實(shí)例方法是定義在元類里面)。該列表不包含父類實(shí)現(xiàn)的方法。outCount
參數(shù)返回方法的個(gè)數(shù)。在獲取到列表后,我們需要使用free()
方法來(lái)釋放它。
**class_replaceMethod**
函數(shù),該函數(shù)的行為可以分為兩種:如果類中不存在name指定的方法,則類似于class_addMethod
函數(shù)一樣會(huì)添加方法;如果類中已存在name指定的方法,則類似于method_setImplementation
一樣替代原方法的實(shí)現(xiàn)。
**class_getMethodImplementation**
函數(shù),該函數(shù)在向類實(shí)例發(fā)送消息時(shí)會(huì)被調(diào)用,并返回一個(gè)指向方法實(shí)現(xiàn)函數(shù)的指針。這個(gè)函數(shù)會(huì)比method_getImplementation(class_getInstanceMethod(cls, name))
更快。返回的函數(shù)指針可能是一個(gè)指向runtime內(nèi)部的函數(shù),而不一定是方法的實(shí)際實(shí)現(xiàn)。例如,如果類實(shí)例無(wú)法響應(yīng)selector
,則返回的函數(shù)指針將是運(yùn)行時(shí)消息轉(zhuǎn)發(fā)機(jī)制的一部分。
**class_respondsToSelector**
函數(shù),我們通常使用NSObject類的respondsToSelector:
或instancesRespondToSelector:
方法來(lái)達(dá)到相同目的。
[](http://southpeak.github.io/2014/10/25/objective-c-runtime-1/#協(xié)議-objc-protocol-list)協(xié)議(objc_protocol_list)
協(xié)議相關(guān)的操作包含以下函數(shù):
<code>
// 添加協(xié)議
BOOL class_addProtocol ( Class cls, Protocol *protocol );
// 返回類是否實(shí)現(xiàn)指定的協(xié)議
BOOL class_conformsToProtocol ( Class cls, Protocol *protocol );
// 返回類實(shí)現(xiàn)的協(xié)議列表
Protocol * class_copyProtocolList ( Class cls, unsigned int *outCount );
class_conformsToProtocol
函數(shù)可以使用NSObject類的conformsToProtocol:
方法來(lái)替代。
class_copyProtocolList
函數(shù)返回的是一個(gè)數(shù)組,在使用后我們需要使用free()
手動(dòng)釋放。
</code>
版本相關(guān)的操作包含以下函數(shù):
<code>
// 獲取版本號(hào)
int class_getVersion ( Class cls );
// 設(shè)置版本號(hào)
void class_setVersion ( Class cls, int version );
</code>
runtime還提供了兩個(gè)函數(shù)來(lái)供CoreFoundation
的tool-free bridging
使用,即:
<code>
Class objc_getFutureClass ( const char *name );
void objc_setFutureClass ( Class cls, const char *name );
</code>
通常我們不直接使用這兩個(gè)函數(shù)。
[](http://southpeak.github.io/2014/10/25/objective-c-runtime-1/#實(shí)例-Example)實(shí)例(Example)
上面列舉了大量類操作的函數(shù),下面我們寫個(gè)實(shí)例,來(lái)看看這些函數(shù)的實(shí)例效果:
<code>
//-----------------------------------------------------------
// MyClass.h
@interface MyClass : NSObject <NSCopying, NSCoding>
@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
@implementation MyClass
+ (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 <objc/runtime.h>
int main(int argc, const char * argv[]) {
@autoreleasepool {
MyClass *myClass = [[MyClass alloc] init];
unsigned int outCount = 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(@"==========================================================");
// 變量實(shí)例大小
NSLog(@"instance size: %zu", class_getInstanceSize(cls));
NSLog(@"==========================================================");
// 成員變量
Ivar *ivars = class_copyIvarList(cls, &outCount);
for (int i = 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 (int i = 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 (int i = 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(@"==========================================================");
// 協(xié)議
Protocol * __unsafe_unretained * protocols = class_copyProtocolList(cls, &outCount);
Protocol * protocol;
for (int i = 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(@"==========================================================");
}
return 0;
}
</code>
這段程序的輸出如下:
<code>
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] ==========================================================
</code>
runtime的強(qiáng)大之處在于它能在運(yùn)行時(shí)創(chuàng)建類和對(duì)象。
**動(dòng)態(tài)創(chuàng)建類涉及到以下幾個(gè)函數(shù):**
// 創(chuàng)建一個(gè)新類和元類
Class objc_allocateClassPair ( Class superclass, const char *name, size_t extraBytes );
// 銷毀一個(gè)類及其相關(guān)聯(lián)的類
void objc_disposeClassPair ( Class cls );
// 在應(yīng)用中注冊(cè)由objc_allocateClassPair創(chuàng)建的類
void objc_registerClassPair ( Class cls );
objc_allocateClassPair
函數(shù):如果我們要?jiǎng)?chuàng)建一個(gè)根類,則superclass
指定為Nil。extraBytes
通常指定為0,該參數(shù)是分配給類和元類對(duì)象尾部的索引ivars
的字節(jié)數(shù)。
為了創(chuàng)建一個(gè)新類,我們需要調(diào)用objc_allocateClassPair
。然后使用諸如class_addMethod
,class_addIvar
等函數(shù)來(lái)為新創(chuàng)建的類添加方法、實(shí)例變量和屬性等。完成這些后,我們需要調(diào)用objc_registerClassPair
函數(shù)來(lái)注冊(cè)類,之后這個(gè)新類就可以在程序中使用了。
實(shí)例方法和實(shí)例變量應(yīng)該添加到類自身上,而類方法應(yīng)該添加到類的元類上。
objc_disposeClassPair
函數(shù)用于銷毀一個(gè)類,不過(guò)需要注意的是,如果程序運(yùn)行中還存在類或其子類的實(shí)例,則不能調(diào)用針對(duì)類調(diào)用該方法。
在前面介紹元類時(shí),我們已經(jīng)有接觸到這幾個(gè)函數(shù)了,在此我們?cè)倥e個(gè)實(shí)例來(lái)看看這幾個(gè)函數(shù)的使用。
<code>
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);
id instance = [[cls alloc] init];
[instance performSelector:@selector(submethod1)];
[instance performSelector:@selector(method1)];
</code>
程序的輸出如下:
<code>
2014-10-23 11:35:31.006 RuntimeTest[3800:66152] run sub method 1
2014-10-23 11:35:31.006 RuntimeTest[3800:66152] run sub method 1
</code>
**動(dòng)態(tài)創(chuàng)建對(duì)象的函數(shù)如下:**
<code>
// 創(chuàng)建類實(shí)例
id class_createInstance ( Class cls, size_t extraBytes );
// 在指定位置創(chuàng)建類實(shí)例
id objc_constructInstance ( Class cls, void *bytes );
// 銷毀類實(shí)例
void * objc_destructInstance ( id obj );
</code>
**class_createInstance**
函數(shù):創(chuàng)建實(shí)例時(shí),會(huì)在默認(rèn)的內(nèi)存區(qū)域?yàn)轭惙峙鋬?nèi)存。extraBytes
參數(shù)表示分配的額外字節(jié)數(shù)。這些額外的字節(jié)可用于存儲(chǔ)在類定義中所定義的實(shí)例變量之外的實(shí)例變量。該函數(shù)在ARC環(huán)境下無(wú)法使用。
調(diào)用class_createInstance
的效果與+alloc
方法類似。不過(guò)在使用class_createInstance
時(shí),我們需要確切的知道我們要用它來(lái)做什么。在下面的例子中,我們用NSString來(lái)測(cè)試一下該函數(shù)的實(shí)際效果:
<code>
id theObject = class_createInstance(NSString.class, sizeof(unsigned));
id str1 = [theObject init];
NSLog(@"%@", [str1 class]);
id str2 = [[NSString alloc] initWithString:@"test"];
NSLog(@"%@", [str2 class]);
</code>
輸出結(jié)果是:
<code>
2014-10-23 12:46:50.781 RuntimeTest[4039:89088] NSString
2014-10-23 12:46:50.781 RuntimeTest[4039:89088] __NSCFConstantString
</code>
可以看到,使用class_createInstance
函數(shù)獲取的是NSString實(shí)例,而不是類簇中的默認(rèn)占位符類__NSCFConstantString
。
**objc_constructInstance**
函數(shù):在指定的位置(bytes)創(chuàng)建類實(shí)例。
**objc_destructInstance**
函數(shù):銷毀一個(gè)類的實(shí)例,但不會(huì)釋放并移除任何與其相關(guān)的引用。
[](http://southpeak.github.io/2014/10/25/objective-c-runtime-1/#實(shí)例操作函數(shù))實(shí)例操作函數(shù)
實(shí)例操作函數(shù)主要是針對(duì)我們創(chuàng)建的實(shí)例對(duì)象的一系列操作函數(shù),我們可以使用這組函數(shù)來(lái)從實(shí)例對(duì)象中獲取我們想要的一些信息,如實(shí)例對(duì)象中變量的值。這組函數(shù)可以分為三小類:
1.針對(duì)整個(gè)對(duì)象進(jìn)行操作的函數(shù),這類函數(shù)包含
// 返回指定對(duì)象的一份拷貝
id object_copy ( id obj, size_t size );
// 釋放指定對(duì)象占用的內(nèi)存
id object_dispose ( id obj );
有這樣一種場(chǎng)景,假設(shè)我們有類A和類B,且類B是類A的子類。類B通過(guò)添加一些額外的屬性來(lái)擴(kuò)展類A?,F(xiàn)在我們創(chuàng)建了一個(gè)A類的實(shí)例對(duì)象,并希望在運(yùn)行時(shí)將這個(gè)對(duì)象轉(zhuǎn)換為B類的實(shí)例對(duì)象,這樣可以添加數(shù)據(jù)到B類的屬性中。這種情況下,我們沒(méi)有辦法直接轉(zhuǎn)換,因?yàn)锽類的實(shí)例會(huì)比A類的實(shí)例更大,沒(méi)有足夠的空間來(lái)放置對(duì)象。此時(shí),我們就要以使用以上幾個(gè)函數(shù)來(lái)處理這種情況,如下代碼所示:
<code>
NSObject *a = [[NSObject alloc] init];
id newB = object_copy(a, class_getInstanceSize(MyClass.class));
object_setClass(newB, MyClass.class);
object_dispose(a);
</code>
2.針對(duì)對(duì)象實(shí)例變量進(jìn)行操作的函數(shù),這類函數(shù)包含:
<code>
// 修改類實(shí)例的實(shí)例變量的值
Ivar object_setInstanceVariable ( id obj, const char *name, void *value );
// 獲取對(duì)象實(shí)例變量的值
Ivar object_getInstanceVariable ( id obj, const char *name, void **outValue );
// 返回指向給定對(duì)象分配的任何額外字節(jié)的指針
void * object_getIndexedIvars ( id obj );
// 返回對(duì)象中實(shí)例變量的值
id object_getIvar ( id obj, Ivar ivar );
// 設(shè)置對(duì)象中實(shí)例變量的值
void object_setIvar ( id obj, Ivar ivar, id value );
</code>
如果實(shí)例變量的Ivar
已經(jīng)知道,那么調(diào)用object_getIvar
會(huì)比object_getInstanceVariable
函數(shù)快,相同情況下,object_setIvar
也比object_setInstanceVariable
快。
3.針對(duì)對(duì)象的類進(jìn)行操作的函數(shù),這類函數(shù)包含:
<code>
// 返回給定對(duì)象的類名
const char * object_getClassName ( id obj );
// 返回對(duì)象的類
Class object_getClass ( id obj );
// 設(shè)置對(duì)象的類
Class object_setClass ( id obj, Class cls );
</code>
[](http://southpeak.github.io/2014/10/25/objective-c-runtime-1/#獲取類定義)獲取類定義
Objective-C動(dòng)態(tài)運(yùn)行庫(kù)會(huì)自動(dòng)注冊(cè)我們代碼中定義的所有的類。我們也可以在運(yùn)行時(shí)創(chuàng)建類定義并使用objc_addClass
函數(shù)來(lái)注冊(cè)它們。runtime
提供了一系列函數(shù)來(lái)獲取類定義相關(guān)的信息,這些函數(shù)主要包括:
<code>
// 獲取已注冊(cè)的類定義的列表
int objc_getClassList ( Class *buffer, int bufferCount );
// 創(chuàng)建并返回一個(gè)指向所有已注冊(cè)類的指針列表
Class * objc_copyClassList ( unsigned int *outCount );
// 返回指定類的類定義
Class objc_lookUpClass ( const char *name );
Class objc_getClass ( const char *name );
Class objc_getRequiredClass ( const char *name );
// 返回指定類的元類
Class objc_getMetaClass ( const char *name );
</code>
objc_getClassList
函數(shù):獲取已注冊(cè)的類定義的列表。我們不能假設(shè)從該函數(shù)中獲取的類對(duì)象是繼承自NSObject體系的,所以在這些類上調(diào)用方法是,都應(yīng)該先檢測(cè)一下這個(gè)方法是否在這個(gè)類中實(shí)現(xiàn)。
下面代碼演示了該函數(shù)的用法:
<code>
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 (int i = 0; i < numClasses; i++) {
Class cls = classes[i];
NSLog(@"class name: %s", class_getName(cls));
}
free(classes);
}
</code>
輸出結(jié)果如下:
<code>
2014-10-23 16:20:52.589 RuntimeTest[8437:188589] number of classes: 1282
2014-10-23 16:20:52.589 RuntimeTest[8437:188589] class name: DDTokenRegexp
2014-10-23 16:20:52.590 RuntimeTest[8437:188589] class name: _NSMostCommonKoreanCharsKeySet
2014-10-23 16:20:52.590 RuntimeTest[8437:188589] class name: OS_xpc_dictionary
2014-10-23 16:20:52.590 RuntimeTest[8437:188589] class name: NSFileCoordinator
2014-10-23 16:20:52.590 RuntimeTest[8437:188589] class name: NSAssertionHandler
2014-10-23 16:20:52.590 RuntimeTest[8437:188589] class name: PFUbiquityTransactionLogMigrator
2014-10-23 16:20:52.591 RuntimeTest[8437:188589] class name: NSNotification
2014-10-23 16:20:52.591 RuntimeTest[8437:188589] class name: NSKeyValueNilSetEnumerator
2014-10-23 16:20:52.591 RuntimeTest[8437:188589] class name: OS_tcp_connection_tls_session
2014-10-23 16:20:52.591 RuntimeTest[8437:188589] class name: _PFRoutines
</code>
......還有大量輸出
獲取類定義的方法有三個(gè):objc_lookUpClass
, objc_getClass
和objc_getRequiredClass
。如果類在運(yùn)行時(shí)未注冊(cè),則objc_lookUpClass
會(huì)返回nil,而objc_getClass
會(huì)調(diào)用類處理回調(diào),并再次確認(rèn)類是否注冊(cè),如果確認(rèn)未注冊(cè),再返回nil。而objc_getRequiredClass
函數(shù)的操作與objc_getClass
相同,只不過(guò)如果沒(méi)有找到類,則會(huì)殺死進(jìn)程。
objc_getMetaClass
函數(shù):如果指定的類沒(méi)有注冊(cè),則該函數(shù)會(huì)調(diào)用類處理回調(diào),并再次確認(rèn)類是否注冊(cè),如果確認(rèn)未注冊(cè),再返回nil。不過(guò),每個(gè)類定義都必須有一個(gè)有效的元類定義,所以這個(gè)函數(shù)總是會(huì)返回一個(gè)元類定義,不管它是否有效。
[](http://southpeak.github.io/2014/10/25/objective-c-runtime-1/#小結(jié))小結(jié)
在這一章中我們介紹了Runtime
運(yùn)行時(shí)中與類和對(duì)象相關(guān)的數(shù)據(jù)結(jié)構(gòu),通過(guò)這些數(shù)據(jù)函數(shù),我們可以管窺Objective-C底層面向?qū)ο髮?shí)現(xiàn)的一些信息。另外,通過(guò)豐富的操作函數(shù),可以靈活地對(duì)這些數(shù)據(jù)進(jìn)行操作。
[](http://southpeak.github.io/2014/10/25/objective-c-runtime-1/#參考)參考
[Objective-C Runtime Reference](https://developer.apple.com/library/ios/documentation/Cocoa/Reference/ObjCRuntimeRef/)
[Objective-C Runtime的數(shù)據(jù)類型](http://www.cnblogs.com/whyandinside/archive/2013/02/26/2933552.html)
[詳解Objective-C的meta-class](http://blog.csdn.net/windyitian/article/details/19810875)
[what are class_setIvarLayout and class_getIvarLayout?](http://stackoverflow.com/questions/16131172/what-are-class-setivarlayout-and-class-getivarlayout)
[What’s the difference between doing alloc and class_createInstance](http://stackoverflow.com/questions/3805499/whats-the-difference-between-doing-alloc-and-class-createinstance)