Objective-C語言是一門動態(tài)語言,他將很多靜態(tài)語言在編譯和鏈接時期做的事情放到了運行時來處理慰于。這種動態(tài)語言的優(yōu)勢在于:我們寫代碼時更具靈活性,如我們可以把消息轉發(fā)給我們想要的對象旁理,或者隨意交換一個方法的實現等。
這種特性意味著objective-c 不僅需要一個編譯器刨肃,還需要一個運行時系統來執(zhí)行編譯的代碼。對于objective-c來說箩帚,這個運行時系統就像一個操作系統一樣:它讓所有的工作可以正常的運行真友。這個運行時系統即Objc Runtime。Objc runtime 其實是一個Runtime庫紧帕,它基本上用c和匯編寫的盔然,這個庫使得c有了面向對象的能力。
Runtime庫主要做下面幾件事:
1是嗜、封裝:在這個庫中愈案,對象可以用C語言中的結構體表示,而方法可以用C函數來實現叠纷,另外再加上了一些額外的特性刻帚。這些結構體和函數被runtime函數封裝后潦嘶,我們就可以在程序運行時創(chuàng)建涩嚣,檢查,修改類掂僵、對象和它們的方法了航厚。
2、找出方法的最終執(zhí)行代碼:當程序執(zhí)行[object doSomething]時锰蓬,會向消息接收者(object)發(fā)送一條消息(doSomething)幔睬,runtime會根據消息接收者是否能響應該消息而做出不同的反應。這將在后面詳細介紹芹扭。
類與對象基礎數據結構
Objective-C類是由Class類型來表示的麻顶,它實際上是一個指向objc_class結構體的指針。它的定義如下:
typedef struct objc_class *Class;
查看objc/runtime.h中objc_class結構體的定義如下:
struct objc_class {
Class isa? OBJC_ISA_AVAILABILITY;
#if !__OBJC2__
Class super_class? ? ? ? ? ? ? ? ? ? ? OBJC2_UNAVAILABLE;? // 父類
const char *name? ? ? ? ? ? ? ? ? ? ? ? OBJC2_UNAVAILABLE;? // 類名
long version? ? ? ? ? ? ? ? ? ? ? ? ? ? OBJC2_UNAVAILABLE;? // 類的版本信息舱卡,默認為0
long info? ? ? ? ? ? ? ? ? ? ? ? ? ? ? OBJC2_UNAVAILABLE;? // 類信息辅肾,供運行期使用的一些位標識
long instance_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_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就會優(yōu)先去
cache中查找泊柬,如果cache沒有椎镣,才去methodLists中查找方法。這樣兽赁,對于那些經常用到的方法的調用状答,但提高了調用的效率。
version:我們可以使用這個字段來提供類的版本信息刀崖。這對于對象的序列化非常有用惊科,它可是讓我們識別出不同類定義版本中實例變量布局的改變。
針對cache亮钦,我們用下面例子來說明其執(zhí)行過程:
NSArray *array = [[NSArray alloc] init];
其流程是:
[NSArray alloc]先被執(zhí)行馆截。因為NSArray沒有+alloc方法,于是去父類NSObject去查找蜂莉。
檢測NSObject是否響應+alloc方法蜡娶,發(fā)現響應,于是檢測NSArray類映穗,并根據其所需的內存空間大小開始分配內存空間窖张,然后把isa指針指向NSArray類。同時男公,+alloc也被加進cache列表里面荤堪。
接著,執(zhí)行-init方法枢赔,如果NSArray響應該方法澄阳,則直接將其加入cache;如果不響應踏拜,則去父類查找碎赢。
在后期的操作中,如果再以[[NSArray alloc] init]這種方式來創(chuàng)建數組速梗,則會直接從cache中取出相應的方法肮塞,直接調用襟齿。
objc_object與id
objc_object是表示一個類的實例的結構體,它的定義如下(objc/objc.h):
struct objc_object {
Class isa? OBJC_ISA_AVAILABILITY;
};
typedef struct objc_object *id;
可以看到枕赵,這個結構體只有一個字體猜欺,即指向其類的isa指針。這樣拷窜,當我們向一個Objective-C對象發(fā)送消息時开皿,運行時庫會根據實例對象的
isa指針找到這個實例對象所屬的類。Runtime庫會在類的方法列表及父類的方法列表中去尋找與消息對應的selector指向的方法篮昧。找到后即運行
這個方法赋荆。
當創(chuàng)建一個特定類的實例對象時,分配的內存包含一個objc_object數據結構懊昨,然后是類的實例變量的數據窄潭。NSObject類的alloc和allocWithZone:方法使用函數class_createInstance來創(chuàng)建objc_object數據結構。
另外還有我們常見的id酵颁,它是一個objc_object結構類型的指針嫉你。它的存在可以讓我們實現類似于C++中泛型的一些操作。該類型的對象可以轉換為任何一種對象材义,有點類似于C語言中void *指針類型的作用均抽。
objc_cache
上面提到了objc_class結構體中的cache字段嫁赏,它用于緩存調用過的方法其掂。這個字段是一個指向objc_cache結構體的指針,其定義如下:
struct objc_cache {
unsigned int mask /* total = mask + 1 */? ? ? ? ? ? ? ? OBJC2_UNAVAILABLE;
unsigned int occupied? ? ? ? ? ? ? ? ? ? ? ? ? ? ? ? ? ? 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可能是不連續(xù)的蝠检。這個數組可能會隨著時間而增長。
元類(Meta Class)
在上面我們提到挚瘟,所有的類自身也是一個對象叹谁,我們可以向這個對象發(fā)送消息(即調用類方法)饲梭。如:
NSArray *array = [NSArray array];
這個例子中,+array消息發(fā)送給了NSArray類焰檩,而這個NSArray也是一個對象憔涉。既然是對象,那么它也是一個objc_object指
針析苫,它包含一個指向其類的一個isa指針监氢。那么這些就有一個問題了,這個isa指針指向什么呢藤违?為了調用+array方法浪腐,這個類的isa指針必須指向一
個包含這些類方法的一個objc_class結構體。這就引出了meta-class的概念
meta-class是一個類對象的類顿乒。
當我們向一個對象發(fā)送消息時议街,runtime會在這個對象所屬的這個類的方法列表中查找方法;而向一個類發(fā)送消息時璧榄,會在這個類的meta-class的方法列表中查找特漩。
meta-class之所以重要,是因為它存儲著一個類的所有類方法骨杂。每個類都會有一個單獨的meta-class哩治,因為每個類的類方法基本不可能完全相同胖秒。
再深入一下,meta-class也是一個類,也可以向它發(fā)送一個消息寻狂,那么它的isa又是指向什么呢龟梦?為了不讓這種結構無限延伸下
去诺祸,Objective-C的設計者讓所有的meta-class的isa指向基類的meta-class旁振,以此作為它們的所屬類。即雳灾,任何
NSObject繼承體系下的meta-class都使用NSObject的meta-class作為自己的所屬類漠酿,而基類的meta-class的
isa指針是指向它自己。這樣就形成了一個完美的閉環(huán)谎亩。
通過上面的描述炒嘲,再加上對objc_class結構體中super_class指針的分析,我們就可以描繪出類及相應meta-class類的一個繼承體系了匈庭,如下圖所示:
對于NSObject繼承體系來說夫凸,其實例方法對體系中的所有實例、類和meta-class都是有效的嚎花;而類方法對于體系內的所有類和meta-class都是有效的寸痢。
講了這么多,我們還是來寫個例子吧:
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
這個例子是在運行時創(chuàng)建了一個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循環(huán)中献烦,我們通過objc_getClass來獲取對象的isa滓窍,并將其打印出來,依此一直回溯到NSObject的meta-class巩那。分析打印結果吏夯,可以看到最后指針指向的地址是0x0,即NSObject的meta-class的類地址即横。
這里需要注意的是:我們在一個類對象調用class方法是無法獲取meta-class噪生,它只是返回類而已。
類與對象操作函數
runtime提供了大量的函數來操作類與對象东囚。類的操作方法大部分是以class為前綴的跺嗽,而對象的操作方法大部分是以objc或object_為前綴。下面我們將根據這些方法的用途來分類討論這些方法的使用页藻。
類相關操作函數
我們可以回過頭去看看objc_class的定義桨嫁,runtime提供的操作類的方法主要就是針對這個結構體中的各個字段的。下面我們分別介紹這一些的函數份帐。并在最后以實例來演示這些函數的具體用法璃吧。
類名(name)
類名操作的函數主要有:
// 獲取類的類名
const char * class_getName ( Class cls );
● 對于class_getName函數,如果傳入的cls為Nil废境,則返回一個字字符串畜挨。
父類(super_class)和元類(meta-class)
父類和元類操作的函數主要有:
// 獲取類的父類
Class class_getSuperclass ( Class cls );
// 判斷給定的Class是否是一個元類
BOOL class_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, 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 );
// 獲取整個成員變量列表
Ivar * class_copyIvarList ( Class cls, unsigned int *outCount );
● class_getInstanceVariable函數钻蔑,它返回一個指向包含name指定的成員變量信息的objc_ivar結構體的指針(Ivar)啥刻。
● class_getClassVariable函數,目前沒有找到關于Objective-C中類變量的信息咪笑,一般認為Objective-C不支持類變量可帽。注意,返回的列表不包含父類的成員變量和屬性窗怒。
●
Objective-C不支持往已存在的類中添加實例變量映跟,因此不管是系統庫提供的提供的類,還是我們自定義的類扬虚,都無法動態(tài)添加成員變量努隙。但如果我們通
過運行時來創(chuàng)建一個類的話,又應該如何給它添加成員變量呢辜昵?這時我們就可以使用class_addIvar函數了剃法。不過需要注意的是,這個方法只能在
objc_allocateClassPair函數與objc_registerClassPair之間調用路鹰。另外贷洲,這個類也不能是元類。成員變量的按字
節(jié)最小對齊量是1<
log2(sizeof(pointer_type))晋柱。
●
class_copyIvarList函數优构,它返回一個指向成員變量信息的數組,數組中每個元素是指向該成員變量信息的objc_ivar結構體的指針雁竞。
這個數組不包含在父類中聲明的變量钦椭。outCount指針返回數組的大小。需要注意的是碑诉,我們必須使用free()來釋放這個數組彪腔。
2.屬性操作函數,主要包含以下函數:
// 獲取指定的屬性
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 );
這一種方法也是針對ivars來操作进栽,不過只操作那些是屬性的值德挣。我們在后面介紹屬性時會再遇到這些函數。
3.在MAC OS X系統中快毛,我們可以使用垃圾回收器格嗅。runtime提供了幾個函數來確定一個對象的內存區(qū)域是否可以被垃圾回收器掃描,以處理strong/weak引用唠帝。這幾個函數定義如下:
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 );
但通常情況下屯掖,我們不需要去主動調用這些方法;在調用objc_registerClassPair時襟衰,會生成合理的布局贴铜。在此不詳細介紹這些函數。
方法(methodLists)
方法操作主要有以下函數:
// 添加方法
BOOL class_addMethod ( Class cls, SEL name, IMP imp, const char *types );
// 獲取實例方法
Method class_getInstanceMethod ( Class cls, SEL name );
// 獲取類方法
Method class_getClassMethod ( Class cls, SEL name );
// 獲取所有方法的數組
Method * class_copyMethodList ( Class cls, unsigned int *outCount );
// 替代方法的實現
IMP class_replaceMethod ( Class cls, SEL name, IMP imp, const char *types );
// 返回方法的具體實現
IMP class_getMethodImplementation ( Class cls, SEL name );
IMP class_getMethodImplementation_stret ( Class cls, SEL name );
// 類實例是否響應指定的selector
BOOL class_respondsToSelector ( Class cls, SEL sel );
●
class_addMethod的實現會覆蓋父類的方法實現,但不會取代本類中已存在的實現绍坝,如果本類中包含一個同名的實現徘意,則函數會返回NO。如果要修
改已存在實現陷嘴,可以使用method_setImplementation映砖。一個Objective-C方法是一個簡單的C函數,它至少包含兩個參數—
self和_cmd灾挨。所以邑退,我們的實現函數(IMP參數指向的函數)至少需要兩個參數,如下所示:
void myMethodIMP(id self, SEL _cmd)
{
// implementation ....
}
與成員變量不同的是劳澄,我們可以為類動態(tài)添加方法地技,不管這個類是否已存在。
另外秒拔,參數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函數版扩,該函數在向類實例發(fā)送消息時會被調用,并返回一個指向方法實現函數的指針侄泽。這個函數會
比method_getImplementation(class_getInstanceMethod(cls,
name))更快礁芦。返回的函數指針可能是一個指向runtime內部的函數,而不一定是方法的實際實現。例如柿扣,如果類實例無法響應selector肖方,則返
回的函數指針將是運行時消息轉發(fā)機制的一部分。
● class_respondsToSelector函數未状,我們通常使用NSObject類的respondsToSelector:或instancesRespondToSelector:方法來達到相同目的俯画。
協議(objc_protocol_list)
協議相關的操作包含以下函數:
// 添加協議
BOOL class_addProtocol ( Class cls, Protocol *protocol );
// 返回類是否實現指定的協議
BOOL class_conformsToProtocol ( Class cls, Protocol *protocol );
// 返回類實現的協議列表
Protocol * class_copyProtocolList ( Class cls, unsigned int *outCount );
● class_conformsToProtocol函數可以使用NSObject類的conformsToProtocol:方法來替代。
● class_copyProtocolList函數返回的是一個數組司草,在使用后我們需要使用free()手動釋放艰垂。
版本(version)
版本相關的操作包含以下函數:
// 獲取版本號
int class_getVersion ( Class cls );
// 設置版本號
void class_setVersion ( Class cls, int version );
其它
runtime還提供了兩個函數來供CoreFoundation的tool-free bridging使用,即:
Class objc_getFutureClass ( const char *name );
void objc_setFutureClass ( Class cls, const char *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
@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
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(@"==========================================================");
// 變量實例大小
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(@"==========================================================");
// 協議
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;
}
這段程序的輸出如下:
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] ==========================================================
動態(tài)創(chuàng)建類和對象
runtime的強大之處在于它能在運行時創(chuàng)建類和對象搔课。
動態(tài)創(chuàng)建類
動態(tài)創(chuàng)建類涉及到以下幾個函數:
// 創(chuàng)建一個新類和元類
Class objc_allocateClassPair ( Class superclass, const char *name, size_t extraBytes );
// 銷毀一個類及其相關聯的類
void objc_disposeClassPair ( Class cls );
// 在應用中注冊由objc_allocateClassPair創(chuàng)建的類
void objc_registerClassPair ( Class cls );
● objc_allocateClassPair函數:如果我們要創(chuàng)建一個根類胰柑,則superclass指定為Nil。extraBytes通常指定為0爬泥,該參數是分配給類和元類對象尾部的索引ivars的字節(jié)數柬讨。
為了創(chuàng)建一個新類,我們需要調用objc_allocateClassPair袍啡。然后使用諸如
class_addMethod踩官,class_addIvar等函數來為新創(chuàng)建的類添加方法、實例變量和屬性等葬馋。完成這些后卖鲤,我們需要調用
objc_registerClassPair函數來注冊類,之后這個新類就可以在程序中使用了畴嘶。
實例方法和實例變量應該添加到類自身上蛋逾,而類方法應該添加到類的元類上。
● objc_disposeClassPair函數用于銷毀一個類窗悯,不過需要注意的是区匣,如果程序運行中還存在類或其子類的實例,則不能調用針對類調用該方法蒋院。
在前面介紹元類時亏钩,我們已經有接觸到這幾個函數了,在此我們再舉個實例來看看這幾個函數的使用欺旧。
Class cls = objc_allocateClassPair(MyClass.class, "MySubClass", 0);
123456789101112131415
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)];
程序的輸出如下:
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
動態(tài)創(chuàng)建對象
動態(tài)創(chuàng)建對象的函數如下:
// 創(chuàng)建類實例
id class_createInstance ( Class cls, size_t extraBytes );
// 在指定位置創(chuàng)建類實例
id objc_constructInstance ( Class cls, void *bytes );
// 銷毀類實例
void * objc_destructInstance ( id obj );
● class_createInstance函數:創(chuàng)建實例時姑丑,會在默認的內存區(qū)域為類分配內存。extraBytes參數表示分配的額外字節(jié)數辞友。這些額外的字節(jié)可用于存儲在類定義中所定義的實例變量之外的實例變量栅哀。該函數在ARC環(huán)境下無法使用震肮。
調用class_createInstance的效果與+alloc方法類似。不過在使用class_createInstance時留拾,我們需要確切的知道我們要用它來做什么戳晌。在下面的例子中,我們用NSString來測試一下該函數的實際效果:
id theObject = class_createInstance(NSString.class, sizeof(unsigned));
123456
id str1 = [theObject init];NSLog(@"%@", [str1 class]);id str2 = [[NSString alloc] initWithString:@"test"];NSLog(@"%@", [str2 class]);
輸出結果是:
2014-10-23 12:46:50.781 RuntimeTest[4039:89088] NSString
2014-10-23 12:46:50.781 RuntimeTest[4039:89088] __NSCFConstantString
可以看到痴柔,使用class_createInstance函數獲取的是NSString實例沦偎,而不是類簇中的默認占位符類__NSCFConstantString。
● objc_constructInstance函數:在指定的位置(bytes)創(chuàng)建類實例咳蔚。
● objc_destructInstance函數:銷毀一個類的實例豪嚎,但不會釋放并移除任何與其相關的引用。
實例操作函數
實例操作函數主要是針對我們創(chuàng)建的實例對象的一系列操作函數屹篓,我們可以使用這組函數來從實例對象中獲取我們想要的一些信息疙渣,如實例對象中變量的值。這組函數可以分為三小類:
1.針對整個對象進行操作的函數堆巧,這類函數包含
// 返回指定對象的一份拷貝
id object_copy ( id obj, size_t size );
// 釋放指定對象占用的內存
id object_dispose ( id obj );
有這樣一種場景妄荔,假設我們有類A和類B,且類B是類A的子類谍肤。類B通過添加一些額外的屬性來擴展類A±沧猓現在我們創(chuàng)建了一個A類的實例對象,并希望在
運行時將這個對象轉換為B類的實例對象荒揣,這樣可以添加數據到B類的屬性中篷角。這種情況下,我們沒有辦法直接轉換系任,因為B類的實例會比A類的實例更大恳蹲,沒有足
夠的空間來放置對象。此時俩滥,我們就要以使用以上幾個函數來處理這種情況嘉蕾,如下代碼所示:
NSObject *a = [[NSObject alloc] init];
id newB = object_copy(a, class_getInstanceSize(MyClass.class));
object_setClass(newB, MyClass.class);
object_dispose(a);
2.針對對象實例變量進行操作的函數,這類函數包含:
// 修改類實例的實例變量的值
Ivar object_setInstanceVariable ( id obj, const char *name, void *value );
// 獲取對象實例變量的值
Ivar object_getInstanceVariable ( id obj, const char *name, void **outValue );
// 返回指向給定對象分配的任何額外字節(jié)的指針
void * object_getIndexedIvars ( id obj );
// 返回對象中實例變量的值
id object_getIvar ( id obj, Ivar ivar );
// 設置對象中實例變量的值
void object_setIvar ( id obj, Ivar ivar, id value );
如果實例變量的Ivar已經知道霜旧,那么調用object_getIvar會比object_getInstanceVariable函數快错忱,相同情況下,object_setIvar也比object_setInstanceVariable快挂据。
3.針對對象的類進行操作的函數以清,這類函數包含:
// 返回給定對象的類名
const char * object_getClassName ( id obj );
// 返回對象的類
Class object_getClass ( id obj );
// 設置對象的類
Class object_setClass ( id obj, Class cls );
獲取類定義
Objective-C動態(tài)運行庫會自動注冊我們代碼中定義的所有的類。我們也可以在運行時創(chuàng)建類定義并使用objc_addClass函數來注冊它們崎逃。runtime提供了一系列函數來獲取類定義相關的信息掷倔,這些函數主要包括:
// 獲取已注冊的類定義的列表
int objc_getClassList ( Class *buffer, int bufferCount );
// 創(chuàng)建并返回一個指向所有已注冊類的指針列表
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 );
● objc_getClassList函數:獲取已注冊的類定義的列表。我們不能假設從該函數中獲取的類對象是繼承自NSObject體系的个绍,所以在這些類上調用方法是今魔,都應該先檢測一下這個方法是否在這個類中實現勺像。
下面代碼演示了該函數的用法:
int numClasses;
Class * classes = NULL;
123456789101112131415
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);}
輸出結果如下:
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
......還有大量輸出
● 獲取類定義的方法有三個:objc_lookUpClass,
objc_getClass和objc_getRequiredClass障贸。如果類在運行時未注冊错森,則objc_lookUpClass會返回nil,而
objc_getClass會調用類處理回調篮洁,并再次確認類是否注冊涩维,如果確認未注冊,再返回nil袁波。而objc_getRequiredClass函數
的操作與objc_getClass相同瓦阐,只不過如果沒有找到類,則會殺死進程篷牌。
● objc_getMetaClass函數:如果指定的類沒有注冊睡蟋,則該函數會調用類處理回調,并再次確認類是否注冊枷颊,如果確認未注冊戳杀,再返回nil。不過夭苗,每個類定義都必須有一個有效的元類定義信卡,所以這個函數總是會返回一個元類定義,不管它是否有效题造。