為了學習Runtime,我們應該需要從幾個方面入手:
- 是什么?
- 為什么會出現?
- 怎么做?
- 分享
- 能干嘛?
是什么? 為什么會出現?
C語言是靜態語言,決定階段是在編譯期,而我們偉大的Apple工程師(其實Objective-C不是Apple發明的,這里姑且這么算吧)把Objective-C定義為了動態語言。那要讓Objectiive-C具有動態語言的特性,就必須有一個東西承載這種特性,而這種特性就是Runtime。個人理解,如有偏頗見諒!
具體其他定義可Google搜索,會看到有很多資料,不多說了。
怎么做?
在Objective-C中一個類其實主要分幾個部分:
- protocol
- Ivar
- property
- method
今天我們就來研究一下這主要的幾個部分。
先上一個隨便寫的類:
#import <Foundation/Foundation.h>
@protocol RuntimeBaseProtocol <NSObject>
@optional
- (void)doBaseAction;
@end
@protocol RuntimeProtocol <NSObject>
@required
- (void)doRequiredAction;
@optional
- (void)doOptionalAction;
@end
@interface RuntimeObject : NSObject <RuntimeProtocol,RuntimeBaseProtocol>
{
NSString *name;
NSString *kind;
}
@property (nonatomic, strong) NSString *value;
@property (nonatomic, assign) int age;
+ (void)doClassMethod;
@end
簡答解釋一下,我們定義了一個叫RuntimeObject的類,類里面定義了一些東西:
- 實現了RuntimeProtocol和RuntimeBaseProtocol協議
- name、kind兩個實例變量
- value、age兩個property
- doClassMethod函數
接下來就是我們今天真正的主題,如何通過Runtime動態性的獲取到這些東西?
其一,我們先Get下Protocol:
unsigned int count;
__unsafe_unretained Protocol **protocols = class_copyProtocolList([RuntimeObject class], &count);
for (unsigned int i = 0; i < count; i++) {
const char *name = protocol_getName(protocols[i]);
printf("%s\n",name);
}
我們發現真的準確的輸出了呃!看:
RuntimeProtocol
RuntimeBaseProtocol
其二,我們再來試試Get下Property:
unsigned int count;
objc_property_t *propertys = class_copyPropertyList([RuntimeObject class], &count);
for (unsigned int i = 0; i < count; i++) {
const char *name = property_getName(propertys[i]);
printf("%s\n",name);
}
輸出如下:
value
age
hash
superclass
description
debugDescription
提示:hash、superclass、description、debugDescription是從NSObject來的
我們再一次
接下來我們Get下Ivar、Method均告完美,哈哈哈!!!
分享
其實我們往上翻翻代碼不難看出,Runtime給我們提供了一組函數,這些函數的形式主要為class_copyXXXX。這組函數就是用來獲取Ivar、Method、Property、Protocol的,他們分別是:
- class_copyIvarList
- class_copyPropertyList
- class_copyProtocolList
- class_copyMethodList
能干嘛?
如果你看到了這些,首先我先感謝你的支持!
?? 但是我們還是得要討論下能用來干嘛呢?有什么鳥用?上面所有的代碼都是在我自己寫的class基礎之上測試的,我都已經有我寫的代碼,何必多此一舉去獲取Method、Property等等吶?
騷年,不急!我們假設哈,如果某天你的主管要求你把他手機上所有安裝過的手機的Bundle ID提取出來,然后用Excel表統計給他,你怎么搞?你是不是想拿榔頭砸他-去你Y的,什么JB玩意。但是騷年,我們不要忘記了,我們是coder啊!我們難道就沒有辦法嗎?
直接上解決方法:
Class LSApplicationWorkspace_class = objc_getClass("LSApplicationWorkspace");
#pragma clang diagnostic push
#pragma clang diagnostic ignored "-Wundeclared-selector"
NSObject *workspace = [LSApplicationWorkspace_class performSelector:@selector(defaultWorkspace)];
NSLog(@"Installed apps:%@",[workspace performSelector:@selector(allApplications)]);
#pragma clang diagnostic pop
你調試一下就知道了,這段代碼可以提取到所有已經安裝過的App列表,包含Bundle ID喔!它就是通過Runtime去獲取workspace類,然后performSelector函數!
后記
好了,Runtime先到這吧。其實它很簡單,也就是Apple提供了一堆函數給你,然后讓你可以操作Ivar、Protocol、Property、Method,也就僅此而已!
附帶:
- 一張截圖
就截這么多吧,剩下的都一樣
:
- 通過Class獲取鏈接庫名稱
Class LSApplicationWorkspace_class = objc_getClass("LSApplicationWorkspace");
const char *name = class_getImageName(LSApplicationWorkspace_class);
printf("%s\n",name);
- 記錄Objective-C中所有消息(也是因為Runtime,新版本SDK已無法直接調用那個函數,但是可通過extern騙過編譯器!)
http://blog.csdn.net/justinjing0612/article/details/9053961
落幕
所有測試代碼如下簡書也不能上傳附件,不上GitHub了,太簡單了
:
{ // Ivar
unsigned int count;
Ivar *ivars = class_copyIvarList([RuntimeObject class], &count);
for (unsigned int i = 0; i < count; i++) {
Ivar ivar = ivars[i];
const char *name = ivar_getName(ivar);
printf("%s\n",name);
}
}
printf("\n\n\n");
{ // property
unsigned int count;
objc_property_t *propertys = class_copyPropertyList([RuntimeObject class], &count);
for (unsigned int i = 0; i < count; i++) {
const char *name = property_getName(propertys[i]);
printf("%s\n",name);
}
}
printf("\n\n\n");
{ // protocol
unsigned int count;
__unsafe_unretained Protocol **protocols = class_copyProtocolList([RuntimeObject class], &count);
for (unsigned int i = 0; i < count; i++) {
const char *name = protocol_getName(protocols[i]);
printf("%s\n",name);
}
}
printf("\n\n\n");
{ // method
unsigned int count;
Method *methods = class_copyMethodList([RuntimeObject class], &count);
for (unsigned int i = 0; i < count; i++) {
SEL sel = method_getName(methods[i]);
printf("%s\n",sel_getName(sel));
}
}
printf("\n\n\n");
{ // Get dynamic framework name
Class LSApplicationWorkspace_class = objc_getClass("LSApplicationWorkspace");
const char *name = class_getImageName(LSApplicationWorkspace_class);
printf("%s\n",name);
}
printf("\n\n\n");
{ // Installed apps
Class LSApplicationWorkspace_class = objc_getClass("LSApplicationWorkspace");
#pragma clang diagnostic push
#pragma clang diagnostic ignored "-Wundeclared-selector"
NSObject *workspace = [LSApplicationWorkspace_class performSelector:@selector(defaultWorkspace)];
NSLog(@"Installed apps:%@",[workspace performSelector:@selector(allApplications)]);
#pragma clang diagnostic pop
}