Objective-C和其他靜態(tài)語言的區(qū)別
- Objective-C將在編譯和鏈接時(shí)期做的事放到運(yùn)行時(shí)來處理
- 即可以在運(yùn)行時(shí)改變其結(jié)構(gòu)
- 新的函數(shù)可以在運(yùn)行時(shí)被引進(jìn)
- 已有的函數(shù)可以被刪除
- 可以交換兩個(gè)方法的實(shí)現(xiàn)等等
1.OC與 Runtime 系統(tǒng)的交互的三種方式
- 通過OC源代碼
- 通過 Foundation 框架的 NSObject 類定義的方法
- -class返回對(duì)象的類启泣;
- -isKindOfClass: 和 -isMemberOfClass: 方法檢查對(duì)象是否存在于指定的類的繼承體系中(是否是其子類或者父類或者當(dāng)前類的成員變量)
- -respondsToSelector: 檢查對(duì)象能否響應(yīng)指定的消息
- -conformsToProtocol:檢查對(duì)象是否實(shí)現(xiàn)了指定協(xié)議類的方法跃闹;
- -methodForSelector: 返回指定方法實(shí)現(xiàn)的地址。
- 3雳窟、通過對(duì) Runtime 庫(kù)函數(shù)的直接調(diào)用
Runtime相關(guān)的術(shù)語及其數(shù)據(jù)結(jié)構(gòu)
1孽水、SEL
- SEL是selector在 Objc 中的表示(Swift 中是 Selector 類)票腰。selector 對(duì)方法名進(jìn)行包裝,以便找到對(duì)應(yīng)的方法實(shí)現(xiàn)女气。它的數(shù)據(jù)結(jié)構(gòu)是
typedef struct objc_selector *SEL;
- 獲取SEL的方式有
- 通過 Objc 編譯器命令@selector()
- 通過Runtime 系統(tǒng)的 sel_registerName 函數(shù)來獲取一個(gè) SEL 類型的方法選擇器杏慰。
2、id
- id 是一個(gè)參數(shù)類型炼鞠,它是指向某個(gè)類的實(shí)例的指針缘滥。定義如下
typedef struct objc_object *id;
struct objc_object {
Class isa;
};
以上定義,看到 objc_object 結(jié)構(gòu)體包含一個(gè) isa 指針谒主,根據(jù) isa 指針就可以找到對(duì)象所屬的類
3朝扼、Class
- 數(shù)據(jù)結(jié)構(gòu)如下:
typedef struct objc_class *Class;
- Class 其實(shí)是指向 objc_class 結(jié)構(gòu)體的指針。objc_class 的數(shù)據(jù)結(jié)構(gòu)如下:
struct objc_class {
Class isa OBJC_ISA_AVAILABILITY;
#if !__OBJC2__
Class super_class (父類指針)
const char *name(類名)
/* 我們可以使用這個(gè)字段來提供類的版本信息霎肯。這對(duì)于對(duì)象的序列化非常有用擎颖,它可是讓我們識(shí)別出不同類定義版本中實(shí)例變量布局的改變。 */
long version(版本)
long info(信息)
long instance_size(類實(shí)例所占內(nèi)存大小)
struct objc_ivar_list *ivars(成員變量列表)
struct objc_method_list **methodLists(方法列表)
struct objc_cache *cache(緩存列表)
struct objc_protocol_list *protocols(協(xié)議列表)
#endif
} OBJC2_UNAVAILABLE;
- struct objc_ivar_list *ivars(成員變量列表)
struct objc_ivar_list {
int ivar_count OBJC2_UNAVAILABLE;
#ifdef __LP64__
int space OBJC2_UNAVAILABLE;
#endif
/* variable length structure */
struct objc_ivar ivar_list[1]
}
- Ivar 是表示成員變量的類型观游。
typedef struct objc_ivar *Ivar;
struct objc_ivar {
char *ivar_name OBJC2_UNAVAILABLE;
char *ivar_type OBJC2_UNAVAILABLE;
int ivar_offset OBJC2_UNAVAILABLE;
#ifdef __LP64__
int space OBJC2_UNAVAILABLE;
#endif
}
- struct objc_method_list **methodLists(方法列表)
struct objc_method_list {
/* 我們可以動(dòng)態(tài)修改 *obsolete 的值來添加成員方法 */
struct objc_method_list *obsolete OBJC2_UNAVAILABLE;
int method_count OBJC2_UNAVAILABLE;
#ifdef __LP64__
int space OBJC2_UNAVAILABLE;
#endif
/* variable length structure */
struct objc_method method_list[1] OBJC2_UNAVAILABLE;
}
- Method 代表類中某個(gè)方法的類型
typedef struct objc_method *Method;
struct objc_method {
/* 方法的編號(hào) */
SEL method_name OBJC2_UNAVAILABLE;
/* 方法類型 method_types 是個(gè) char 指針搂捧,存儲(chǔ)方法的參數(shù)類型和返回值類型 */
char *method_types OBJC2_UNAVAILABLE;
/* 指向了方法的實(shí)現(xiàn),本質(zhì)是一個(gè)函數(shù)指針 */
IMP method_imp OBJC2_UNAVAILABLE;
}
- IMP結(jié)構(gòu)
typedef id (*IMP)(id, SEL, ...);
它就是一個(gè)函數(shù)指針懂缕,這是由編譯器生成的允跑。當(dāng)你發(fā)起一個(gè) ObjC 消息之后,最終它會(huì)執(zhí)行的那段代碼,就是由這個(gè)函數(shù)指針指定的聋丝。而 IMP 這個(gè)函數(shù)指針就指向了這個(gè)方法的實(shí)現(xiàn)荤崇。
你會(huì)發(fā)現(xiàn) IMP 指向的方法與 objc_msgSend 函數(shù)類型相同,參數(shù)都包含 id 和 SEL 類型潮针。每個(gè)方法名都對(duì)應(yīng)一個(gè) SEL 類型的方法選擇器,而每個(gè)實(shí)例對(duì)象中的 SEL 對(duì)應(yīng)的方法實(shí)現(xiàn)肯定是唯一的倚喂,通過一組 id和 SEL 參數(shù)就能確定唯一的方法實(shí)現(xiàn)地址每篷。
- Cache 定義如下
typedef struct objc_cache *Cache
struct objc_cache {
unsigned int mask /* total = mask + 1 */ OBJC2_UNAVAILABLE;
unsigned int occupied OBJC2_UNAVAILABLE;
Method buckets[1] OBJC2_UNAVAILABLE;
};
- Property定義如下
typedef struct objc_property *Property;
typedef struct objc_property *objc_property_t;//這個(gè)更常用
- 可以以通過class_copyPropertyList 方法獲取類中的屬性
objc_property_t *class_copyPropertyList(Class cls, unsigned int *outCount)
- 返回的是屬性列表,列表中每個(gè)元素都是一個(gè) objc_property_t 指針,方法使用示例如下:
#import <Foundation/Foundation.h>
@interface Person : NSObject
/** 姓名 */
@property (strong, nonatomic) NSString *name;
/** age */
@property (assign, nonatomic) int age;
/** weight */
@property (assign, nonatomic) double weight;
@end
以上是一個(gè) Person 類端圈,有3個(gè)屬性焦读。讓我們用上述方法獲取類的運(yùn)行時(shí)屬性
unsigned int outCount = 0;
objc_property_t *properties = class_copyPropertyList([Person class], &outCount);
NSLog(@"%d", outCount);
for (NSInteger i = 0; i < outCount; i++) {
NSString *name = @(property_getName(properties[i]));
NSString *attributes = @(property_getAttributes(properties[i]));
NSLog(@"%@--------%@", name, attributes);
}
打印結(jié)果如下:
2016-12-17 11:27:28.473 test[2321:451525] 3
2016-12-17 11:27:28.473 test[2321:451525] name--------T@"NSString",&,N,V_name
2016-12-17 11:27:28.473 test[2321:451525] age--------Ti,N,V_age
2016-12-17 11:27:28.474 test[2321:451525] weight--------Td,N,V_weight
property_getName 用來查找屬性的名稱,返回 c 字符串舱权。property_getAttributes 函數(shù)挖掘?qū)傩缘恼鎸?shí)名稱和 @encode 類型矗晃,返回 c 字符串
- 可以以通過protocol_copyPropertyList 方法獲取協(xié)議中的屬性
objc_property_t *protocol_copyPropertyList(Protocol *proto, unsigned int *outCount)