iOS 開發(fā)中 runtime 常用的幾種方法

公司項(xiàng)目中用了一些 runtime 相關(guān)的知識, 初看時(shí)有些蒙, 雖然用的并不多, 但還是想著系統(tǒng)的把 runtime 相關(guān)的常用方法整理一下, 自己以后用著方便, 也希望對看到的朋友有所幫助.

一、runtime 簡介

runtime 簡稱運(yùn)行時(shí)眠蚂,是系統(tǒng)在運(yùn)行的時(shí)候的一些機(jī)制驹止,其中最主要的是消息機(jī)制萍诱。它是一套比較底層的純 C 語言 API, 屬于一個(gè) C 語言庫,包含了很多底層的 C 語言 API。我們平時(shí)編寫的 OC 代碼,在程序運(yùn)行過程時(shí)柿究,其實(shí)最終都是轉(zhuǎn)成了 runtime 的 C 語言代碼。如下所示:

// OC代碼:
[Person coding];

//運(yùn)行時(shí) runtime 會將它轉(zhuǎn)化成 C 語言的代碼:
objc_msgSend(Person, @selector(coding));

二黄选、相關(guān)函數(shù)

// 遍歷某個(gè)類所有的成員變量
class_copyIvarList

// 遍歷某個(gè)類所有的方法
class_copyMethodList

// 獲取指定名稱的成員變量
class_getInstanceVariable

// 獲取成員變量名
ivar_getName

// 獲取成員變量類型編碼
ivar_getTypeEncoding

// 獲取某個(gè)對象成員變量的值
object_getIvar

// 設(shè)置某個(gè)對象成員變量的值
object_setIvar

// 給對象發(fā)送消息
objc_msgSend

三蝇摸、相關(guān)應(yīng)用

  • 更改屬性值
  • 動(dòng)態(tài)添加屬性
  • 動(dòng)態(tài)添加方法
  • 交換方法的實(shí)現(xiàn)
  • 攔截并替換方法
  • 在方法上增加額外功能
  • 歸檔解檔
  • 字典轉(zhuǎn)模型

以上八種用法用代碼都實(shí)現(xiàn)了, 文末會貼出代碼地址.


runtime

四、代碼實(shí)現(xiàn)

要使用runtime,要先引入頭文件#import <objc/runtime.h>

4.1 更改屬性值

用 runtime 修改一個(gè)對象的屬性值

    unsigned int count = 0;
    // 動(dòng)態(tài)獲取類中的所有屬性(包括私有)
    Ivar *ivar = class_copyIvarList(_person.class, &count);
    // 遍歷屬性找到對應(yīng)字段
    for (int i = 0; i < count; i ++) {
        Ivar tempIvar = ivar[i];
        const char *varChar = ivar_getName(tempIvar);
        NSString *varString = [NSString stringWithUTF8String:varChar];
        if ([varString isEqualToString:@"_name"]) {
            // 修改對應(yīng)的字段值
            object_setIvar(_person, tempIvar, @"更改屬性值成功");
            break;
        }
    }

4.2 動(dòng)態(tài)添加屬性

用 runtime 為一個(gè)類添加屬性, iOS 分類里一般會這樣用, 我們建立一個(gè)分類, NSObject+NNAddAttribute.h, 并添加以下代碼:

- (void)setName:(NSString *)name {
    objc_setAssociatedObject(self, @"name", name, OBJC_ASSOCIATION_RETAIN_NONATOMIC);
}

- (NSString *)name {
    return objc_getAssociatedObject(self, @"name");
}

這樣只要引用 NSObject+NNAddAttribute.h, 用 NSObject 創(chuàng)建的對象就會有一個(gè) name 屬性, 我們可以直接這樣寫:

    NSObject *person = [NSObject new];
    person.name = @"以夢為馬";

4.3 動(dòng)態(tài)添加方法

person 類中沒有 coding 方法,我們用 runtime 給 person 類添加了一個(gè)名字叫 coding 的方法搭伤,最終再調(diào)用coding方法做出相應(yīng). 下面代碼的幾個(gè)參數(shù)需要注意一下:

- (void)buttonClick:(UIButton *)sender {
    /*
     動(dòng)態(tài)添加 coding 方法
     (IMP)codingOC 意思是 codingOC 的地址指針;
     "v@:" 意思是,v 代表無返回值 void啡专,如果是 i 則代表 int;@代表 id sel; : 代表 SEL _cmd;
     “v@:@@” 意思是制圈,兩個(gè)參數(shù)的沒有返回值们童。
     */
    class_addMethod([_person class], @selector(coding), (IMP)codingOC, "v@:");
    // 調(diào)用 coding 方法響應(yīng)事件
    if ([_person respondsToSelector:@selector(coding)]) {
        [_person performSelector:@selector(coding)];
        self.testLabelText = @"添加方法成功";
    } else {
        self.testLabelText = @"添加方法失敗";
    }
}

// 編寫 codingOC 的實(shí)現(xiàn)
void codingOC(id self,SEL _cmd) {
    NSLog(@"添加方法成功");
}

4.4 交換方法的實(shí)現(xiàn)

某個(gè)類有兩個(gè)方法, 比如 person 類有兩個(gè)方法, coding 方法與 eating 方法, 我們用 runtime 交換一下這兩個(gè)方法, 就會出現(xiàn)這樣的情況, 當(dāng)我們調(diào)用 coding 的時(shí)候, 執(zhí)行的是 eating, 當(dāng)我們調(diào)用 eating 的時(shí)候, 執(zhí)行的是 coding, 如下面的動(dòng)態(tài)效果圖.

    Method oriMethod = class_getInstanceMethod(_person.class, @selector(coding));
    Method curMethod = class_getInstanceMethod(_person.class, @selector(eating));
    method_exchangeImplementations(oriMethod, curMethod);
交換方法的實(shí)現(xiàn)

4.5 攔截并替換方法

這個(gè)功能和上面的其實(shí)有些類似, 攔截并替換方法可以攔截并替換同一個(gè)類的, 也可以在兩個(gè)類之間進(jìn)行, 我這里用了兩個(gè)不同的類, 下面是簡單的代碼實(shí)現(xiàn).

    _person = [NNPerson new];
    _library = [NNLibrary new];
    self.testLabelText = [_library libraryMethod];
    Method oriMethod = class_getInstanceMethod(_person.class, @selector(changeMethod));
    Method curMethod = class_getInstanceMethod(_library.class, @selector(libraryMethod));
    method_exchangeImplementations(oriMethod, curMethod);

4.6 在方法上增加額外功能

這個(gè)使用場景還是挺多的, 比如我們需要記錄 APP 中某一個(gè)按鈕的點(diǎn)擊次數(shù), 這個(gè)時(shí)候我們便可以利用 runtime 來實(shí)現(xiàn)這個(gè)功能. 我這里寫了個(gè) UIButton 的子類, 然后在 + (void)load 中用 runtime 給它增加了一個(gè)功能, 核心代碼及實(shí)現(xiàn)效果圖如下:

+ (void)load {
    static dispatch_once_t onceToken;
    dispatch_once(&onceToken, ^{
        Method oriMethod = class_getInstanceMethod(self.class, @selector(sendAction:to:forEvent:));
        Method cusMethod = class_getInstanceMethod(self.class, @selector(customSendAction:to:forEvent:));
        // 判斷自定義的方法是否實(shí)現(xiàn), 避免崩潰
        BOOL addSuccess = class_addMethod(self.class, @selector(sendAction:to:forEvent:), method_getImplementation(cusMethod), method_getTypeEncoding(cusMethod));
        if (addSuccess) {
            // 沒有實(shí)現(xiàn), 將源方法的實(shí)現(xiàn)替換到交換方法的實(shí)現(xiàn)
            class_replaceMethod(self.class, @selector(customSendAction:to:forEvent:), method_getImplementation(oriMethod), method_getTypeEncoding(oriMethod));
        } else {
            // 已經(jīng)實(shí)現(xiàn), 直接交換方法
            method_exchangeImplementations(oriMethod, cusMethod);
        }
    });
}
在方法上增加額外功能

4.7 歸檔解檔

當(dāng)我們使用 NSCoding 進(jìn)行歸檔及解檔時(shí), 如果不用 runtime, 那么不管模型里面有多少屬性, 我們都需要對其實(shí)現(xiàn)一遍 encodeObjectdecodeObjectForKey 方法, 如果模型里面有 10000 個(gè)屬性, 那么我們就需要寫 10000 句encodeObjectdecodeObjectForKey 方法, 這個(gè)時(shí)候用 runtime, 便可以充分體驗(yàn)其好處(以下只是核心代碼, 具體代碼請見 demo).

- (void)encodeWithCoder:(NSCoder *)aCoder {
    unsigned int count = 0;
    // 獲取類中所有屬性
    Ivar *ivars = class_copyIvarList(self.class, &count);
    // 遍歷屬性
    for (int i = 0; i < count; i ++) {
        // 取出 i 位置對應(yīng)的屬性
        Ivar ivar = ivars[i];
        // 查看屬性
        const char *name = ivar_getName(ivar);
        NSString *key = [NSString stringWithUTF8String:name];
        // 利用 KVC 進(jìn)行取值,根據(jù)屬性名稱獲取對應(yīng)的值
        id value = [self valueForKey:key];
        [aCoder encodeObject:value forKey:key];
    }
    free(ivars);
}

- (instancetype)initWithCoder:(NSCoder *)aDecoder {
    if (self = [super init]) {
        unsigned int count = 0;
        // 獲取類中所有屬性
        Ivar *ivars = class_copyIvarList(self.class, &count);
        // 遍歷屬性
        for (int i = 0; i < count; i ++) {
            // 取出 i 位置對應(yīng)的屬性
            Ivar ivar = ivars[i];
            // 查看屬性
            const char *name = ivar_getName(ivar);
            NSString *key = [NSString stringWithUTF8String:name];
            // 進(jìn)行解檔取值
            id value = [aDecoder decodeObjectForKey:key];
            // 利用 KVC 對屬性賦值
            [self setValue:value forKey:key];
        }
    }
    return self;
}

4.8 字典轉(zhuǎn)模型

字典轉(zhuǎn)模型我們通常用的都是第三方, MJExtension, YYModel 等, 但也有必要了解一下其實(shí)現(xiàn)方式: 遍歷模型中的所有屬性鲸鹦,根據(jù)模型的屬性名慧库,去字典中查找對應(yīng)的 key,取出對應(yīng)的值馋嗜,給模型的屬性賦值完沪。

/** 字典轉(zhuǎn)模型 **/
+ (instancetype)modelWithDict:(NSDictionary *)dict {
    id objc = [[self alloc] init];
    unsigned int count = 0;
    // 獲取成員屬性數(shù)組
    Ivar *ivarList = class_copyIvarList(self, &count);
    // 遍歷所有的成員屬性名
    for (int i = 0; i < count; i ++) {
        // 獲取成員屬性
        Ivar ivar = ivarList[i];
        // 獲取成員屬性名
        NSString *ivarName = [NSString stringWithUTF8String:ivar_getName(ivar)];
        NSString *key = [ivarName substringFromIndex:1];
        // 從字典中取出對應(yīng) value 給模型屬性賦值
        id value = dict[key];
        // 獲取成員屬性類型
        NSString *ivarType = [NSString stringWithUTF8String:ivar_getTypeEncoding(ivar)];
        // 判斷 value 是不是字典
        if ([value isKindOfClass:[NSDictionary class]]) {
            ivarType = [ivarType stringByReplacingOccurrencesOfString:@"@" withString:@""];
            ivarType = [ivarType stringByReplacingOccurrencesOfString:@"\"" withString:@""];
            Class modalClass = NSClassFromString(ivarType);
            // 字典轉(zhuǎn)模型
            if (modalClass) {
                // 字典轉(zhuǎn)模型
                value = [modalClass modelWithDict:value];
            }
        }
        if ([value isKindOfClass:[NSArray class]]) {
            // 判斷對應(yīng)類有沒有實(shí)現(xiàn)字典數(shù)組轉(zhuǎn)模型數(shù)組的協(xié)議
            if ([self respondsToSelector:@selector(arrayContainModelClass)]) {
                // 轉(zhuǎn)換成id類型,就能調(diào)用任何對象的方法
                id idSelf = self;
                // 獲取數(shù)組中字典對應(yīng)的模型
                NSString *type = [idSelf arrayContainModelClass][key];
                // 生成模型
                Class classModel = NSClassFromString(type);
                NSMutableArray *arrM = [NSMutableArray array];
                // 遍歷字典數(shù)組嵌戈,生成模型數(shù)組
                for (NSDictionary *dict in value) {
                    // 字典轉(zhuǎn)模型
                    id model =  [classModel modelWithDict:dict];
                    [arrM addObject:model];
                }
                // 把模型數(shù)組賦值給value
                value = arrM;
            }
        }
        // KVC 字典轉(zhuǎn)模型
        if (value) {
            [objc setValue:value forKey:key];
        }
    }
    return objc;
}

上面的所有代碼都可以在這里下載: runtime 練習(xí): NNRuntimeTest

最后編輯于
?著作權(quán)歸作者所有,轉(zhuǎn)載或內(nèi)容合作請聯(lián)系作者
  • 序言:七十年代末,一起剝皮案震驚了整個(gè)濱河市听皿,隨后出現(xiàn)的幾起案子熟呛,更是在濱河造成了極大的恐慌,老刑警劉巖尉姨,帶你破解...
    沈念sama閱讀 221,198評論 6 514
  • 序言:濱河連續(xù)發(fā)生了三起死亡事件庵朝,死亡現(xiàn)場離奇詭異,居然都是意外死亡又厉,警方通過查閱死者的電腦和手機(jī)九府,發(fā)現(xiàn)死者居然都...
    沈念sama閱讀 94,334評論 3 398
  • 文/潘曉璐 我一進(jìn)店門,熙熙樓的掌柜王于貴愁眉苦臉地迎上來覆致,“玉大人侄旬,你說我怎么就攤上這事』吐瑁” “怎么了儡羔?”我有些...
    開封第一講書人閱讀 167,643評論 0 360
  • 文/不壞的土叔 我叫張陵宣羊,是天一觀的道長。 經(jīng)常有香客問我汰蜘,道長仇冯,這世上最難降的妖魔是什么? 我笑而不...
    開封第一講書人閱讀 59,495評論 1 296
  • 正文 為了忘掉前任族操,我火速辦了婚禮苛坚,結(jié)果婚禮上,老公的妹妹穿的比我還像新娘色难。我一直安慰自己泼舱,他們只是感情好,可當(dāng)我...
    茶點(diǎn)故事閱讀 68,502評論 6 397
  • 文/花漫 我一把揭開白布莱预。 她就那樣靜靜地躺著柠掂,像睡著了一般。 火紅的嫁衣襯著肌膚如雪依沮。 梳的紋絲不亂的頭發(fā)上涯贞,一...
    開封第一講書人閱讀 52,156評論 1 308
  • 那天,我揣著相機(jī)與錄音危喉,去河邊找鬼宋渔。 笑死,一個(gè)胖子當(dāng)著我的面吹牛辜限,可吹牛的內(nèi)容都是我干的皇拣。 我是一名探鬼主播,決...
    沈念sama閱讀 40,743評論 3 421
  • 文/蒼蘭香墨 我猛地睜開眼薄嫡,長吁一口氣:“原來是場噩夢啊……” “哼氧急!你這毒婦竟也來了?” 一聲冷哼從身側(cè)響起毫深,我...
    開封第一講書人閱讀 39,659評論 0 276
  • 序言:老撾萬榮一對情侶失蹤吩坝,失蹤者是張志新(化名)和其女友劉穎,沒想到半個(gè)月后哑蔫,有當(dāng)?shù)厝嗽跇淞掷锇l(fā)現(xiàn)了一具尸體钉寝,經(jīng)...
    沈念sama閱讀 46,200評論 1 319
  • 正文 獨(dú)居荒郊野嶺守林人離奇死亡,尸身上長有42處帶血的膿包…… 初始之章·張勛 以下內(nèi)容為張勛視角 年9月15日...
    茶點(diǎn)故事閱讀 38,282評論 3 340
  • 正文 我和宋清朗相戀三年闸迷,在試婚紗的時(shí)候發(fā)現(xiàn)自己被綠了嵌纲。 大學(xué)時(shí)的朋友給我發(fā)了我未婚夫和他白月光在一起吃飯的照片。...
    茶點(diǎn)故事閱讀 40,424評論 1 352
  • 序言:一個(gè)原本活蹦亂跳的男人離奇死亡腥沽,死狀恐怖逮走,靈堂內(nèi)的尸體忽然破棺而出,到底是詐尸還是另有隱情今阳,我是刑警寧澤言沐,帶...
    沈念sama閱讀 36,107評論 5 349
  • 正文 年R本政府宣布邓嘹,位于F島的核電站,受9級特大地震影響险胰,放射性物質(zhì)發(fā)生泄漏汹押。R本人自食惡果不足惜,卻給世界環(huán)境...
    茶點(diǎn)故事閱讀 41,789評論 3 333
  • 文/蒙蒙 一起便、第九天 我趴在偏房一處隱蔽的房頂上張望棚贾。 院中可真熱鬧,春花似錦榆综、人聲如沸妙痹。這莊子的主人今日做“春日...
    開封第一講書人閱讀 32,264評論 0 23
  • 文/蒼蘭香墨 我抬頭看了看天上的太陽怯伊。三九已至,卻和暖如春判沟,著一層夾襖步出監(jiān)牢的瞬間耿芹,已是汗流浹背。 一陣腳步聲響...
    開封第一講書人閱讀 33,390評論 1 271
  • 我被黑心中介騙來泰國打工挪哄, 沒想到剛下飛機(jī)就差點(diǎn)兒被人妖公主榨干…… 1. 我叫王不留吧秕,地道東北人。 一個(gè)月前我還...
    沈念sama閱讀 48,798評論 3 376
  • 正文 我出身青樓迹炼,卻偏偏與公主長得像砸彬,于是被迫代替她去往敵國和親。 傳聞我的和親對象是個(gè)殘疾皇子斯入,可洞房花燭夜當(dāng)晚...
    茶點(diǎn)故事閱讀 45,435評論 2 359

推薦閱讀更多精彩內(nèi)容

  • 1.ios高性能編程 (1).內(nèi)層 最小的內(nèi)層平均值和峰值(2).耗電量 高效的算法和數(shù)據(jù)結(jié)構(gòu)(3).初始化時(shí)...
    歐辰_OSR閱讀 29,405評論 8 265
  • 1砂碉、通過CocoaPods安裝項(xiàng)目名稱項(xiàng)目信息 AFNetworking網(wǎng)絡(luò)請求組件 FMDB本地?cái)?shù)據(jù)庫組件 SD...
    陽明先生_X自主閱讀 15,985評論 3 119
  • 這四年本該從外企退身回家做個(gè)安詳?shù)闹鲖D,好好相夫教子的刻两,結(jié)果一不小心被趨勢推進(jìn)了微商行列增蹭,一干就是四年! 從小就爭...
    chicashley閱讀 364評論 0 1
  • 有沒有人曾經(jīng)說過 她想和你 在冬日里的下午 圍坐在火爐邊 喝熱氣騰騰的茶 講溫暖的故事 分享好看的書 有沒有人曾經(jīng)...
    懷澍閱讀 360評論 2 4
  • 記得第一次去參加華東區(qū)架構(gòu)師同學(xué)會的時(shí)候闹伪,針對“高原奶爸”宇成兄的項(xiàng)目,簡單做了項(xiàng)目梳理的方法小結(jié)壮池,后來也被很多同...
    子斌老師聊眾籌閱讀 1,162評論 0 2