Runtime實戰

為了學習Runtime,我們應該需要從幾個方面入手:

  1. 是什么?
  2. 為什么會出現?
  3. 怎么做?
  4. 分享
  5. 能干嘛?

是什么? 為什么會出現?

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
timg.jpeg

其二,我們再來試試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來的

我們再一次


timg.jpeg

接下來我們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,也就僅此而已!

附帶:

  • 一張截圖就截這么多吧,剩下的都一樣
Runtime_1.jpg
  • 通過Class獲取鏈接庫名稱
Class LSApplicationWorkspace_class = objc_getClass("LSApplicationWorkspace");
const char *name = class_getImageName(LSApplicationWorkspace_class);
printf("%s\n",name);

落幕

所有測試代碼如下簡書也不能上傳附件,不上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
        }
最后編輯于
?著作權歸作者所有,轉載或內容合作請聯系作者
平臺聲明:文章內容(如有圖片或視頻亦包括在內)由作者上傳并發布,文章內容僅代表作者本人觀點,簡書系信息發布平臺,僅提供信息存儲服務。

推薦閱讀更多精彩內容

  • 轉至元數據結尾創建: 董瀟偉,最新修改于: 十二月 23, 2016 轉至元數據起始第一章:isa和Class一....
    40c0490e5268閱讀 1,757評論 0 9
  • 我們常常會聽說 Objective-C 是一門動態語言,那么這個「動態」表現在哪呢?我想最主要的表現就是 Obje...
    Ethan_Struggle閱讀 2,230評論 0 7
  • 本文轉載自:http://southpeak.github.io/2014/10/25/objective-c-r...
    idiot_lin閱讀 948評論 0 4
  • Objective-C語言是一門動態語言,他將很多靜態語言在編譯和鏈接時期做的事情放到了運行時來處理。這種動態語言...
    tigger丨閱讀 1,426評論 0 8
  • 這篇文章完全是基于南峰子老師博客的轉載 這篇文章完全是基于南峰子老師博客的轉載 這篇文章完全是基于南峰子老師博客的...
    西木閱讀 30,585評論 33 466