亚洲香蕉成人av网站在线观看_欧美精品成人91久久久久久久_久久久久久久久久久亚洲_热久久视久久精品18亚洲精品_国产精自产拍久久久久久_亚洲色图国产精品_91精品国产网站_中文字幕欧美日韩精品_国产精品久久久久久亚洲调教_国产精品久久一区_性夜试看影院91社区_97在线观看视频国产_68精品久久久久久欧美_欧美精品在线观看_国产精品一区二区久久精品_欧美老女人bb

首頁 > 系統 > iOS > 正文

iOS開發中runtime常用的幾種方法示例總結

2019-10-21 18:40:03
字體:
來源:轉載
供稿:網友

前言

Objective-C runtime是一個實現Objective-C語言的C庫。它是一門編譯型語言、也是一門動態型的語言(這里強調下OC是靜態類型語言),之前沒接觸runtime的時候也不覺著它有多重要,接觸之后才發現其實runtime挺強大的。就拿我們在iOS開發中所使用的OC編程語言來講,OC之所以能夠做到即是編譯型語言,又能做到動態語言,就是得益于runtime的機制。

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

一、runtime 簡介

runtime 簡稱運行時,是系統在運行的時候的一些機制,其中最主要的是消息機制。它是一套比較底層的純 C 語言 API, 屬于一個 C 語言庫,包含了很多底層的 C 語言 API。我們平時編寫的 OC 代碼,在程序運行過程時,其實最終都是轉成了 runtime 的 C 語言代碼。如下所示:

// OC代碼:[Person coding];//運行時 runtime 會將它轉化成 C 語言的代碼:objc_msgSend(Person, @selector(coding));

二、相關函數

// 遍歷某個類所有的成員變量class_copyIvarList// 遍歷某個類所有的方法class_copyMethodList// 獲取指定名稱的成員變量class_getInstanceVariable// 獲取成員變量名ivar_getName// 獲取成員變量類型編碼ivar_getTypeEncoding// 獲取某個對象成員變量的值object_getIvar// 設置某個對象成員變量的值object_setIvar// 給對象發送消息objc_msgSend

三、相關應用

  • 更改屬性值
  • 動態添加屬性
  • 動態添加方法
  • 交換方法的實現
  • 攔截并替換方法
  • 在方法上增加額外功能
  • 歸檔解檔
  • 字典轉模型

以上八種用法用代碼都實現了, 文末會貼出代碼地址.

iOS開發,runtime
runtime

四、代碼實現

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

4.1 更改屬性值

用 runtime 修改一個對象的屬性值

 unsigned int count = 0; // 動態獲取類中的所有屬性(包括私有) Ivar *ivar = class_copyIvarList(_person.class, &count); // 遍歷屬性找到對應字段 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"]) {   // 修改對應的字段值   object_setIvar(_person, tempIvar, @"更改屬性值成功");   break;  } }

4.2 動態添加屬性

用 runtime 為一個類添加屬性, iOS 分類里一般會這樣用, 我們建立一個分類, 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 創建的對象就會有一個 name 屬性, 我們可以直接這樣寫:

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

4.3 動態添加方法

person 類中沒有 coding 方法,我們用 runtime 給 person 類添加了一個名字叫 coding 的方法,最終再調用coding方法做出相應. 下面代碼的幾個參數需要注意一下:

- (void)buttonClick:(UIButton *)sender { /*  動態添加 coding 方法  (IMP)codingOC 意思是 codingOC 的地址指針;  "v@:" 意思是,v 代表無返回值 void,如果是 i 則代表 int;@代表 id sel; : 代表 SEL _cmd;  “v@:@@” 意思是,兩個參數的沒有返回值。  */ class_addMethod([_person class], @selector(coding), (IMP)codingOC, "v@:"); // 調用 coding 方法響應事件 if ([_person respondsToSelector:@selector(coding)]) {  [_person performSelector:@selector(coding)];  self.testLabelText = @"添加方法成功"; } else {  self.testLabelText = @"添加方法失敗"; }} // 編寫 codingOC 的實現void codingOC(id self,SEL _cmd) { NSLog(@"添加方法成功");}

4.4 交換方法的實現

某個類有兩個方法, 比如 person 類有兩個方法, coding 方法與 eating 方法, 我們用 runtime 交換一下這兩個方法, 就會出現這樣的情況, 當我們調用 coding 的時候, 執行的是 eating, 當我們調用 eating 的時候, 執行的是 coding, 如下面的動態效果圖.

 Method oriMethod = class_getInstanceMethod(_person.class, @selector(coding)); Method curMethod = class_getInstanceMethod(_person.class, @selector(eating)); method_exchangeImplementations(oriMethod, curMethod);

iOS開發,runtime

交換方法的實現

4.5 攔截并替換方法

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

 _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 在方法上增加額外功能

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

+ (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:));  // 判斷自定義的方法是否實現, 避免崩潰  BOOL addSuccess = class_addMethod(self.class, @selector(sendAction:to:forEvent:), method_getImplementation(cusMethod), method_getTypeEncoding(cusMethod));  if (addSuccess) {   // 沒有實現, 將源方法的實現替換到交換方法的實現   class_replaceMethod(self.class, @selector(customSendAction:to:forEvent:), method_getImplementation(oriMethod), method_getTypeEncoding(oriMethod));  } else {   // 已經實現, 直接交換方法   method_exchangeImplementations(oriMethod, cusMethod);  } });}

iOS開發,runtime

在方法上增加額外功能

4.7 歸檔解檔

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

- (void)encodeWithCoder:(NSCoder *)aCoder { unsigned int count = 0; // 獲取類中所有屬性 Ivar *ivars = class_copyIvarList(self.class, &count); // 遍歷屬性 for (int i = 0; i < count; i ++) {  // 取出 i 位置對應的屬性  Ivar ivar = ivars[i];  // 查看屬性  const char *name = ivar_getName(ivar);  NSString *key = [NSString stringWithUTF8String:name];  // 利用 KVC 進行取值,根據屬性名稱獲取對應的值  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 位置對應的屬性   Ivar ivar = ivars[i];   // 查看屬性   const char *name = ivar_getName(ivar);   NSString *key = [NSString stringWithUTF8String:name];   // 進行解檔取值   id value = [aDecoder decodeObjectForKey:key];   // 利用 KVC 對屬性賦值   [self setValue:value forKey:key];  } } return self;}

4.8 字典轉模型

字典轉模型我們通常用的都是第三方, MJExtension, YYModel 等, 但也有必要了解一下其實現方式: 遍歷模型中的所有屬性,根據模型的屬性名,去字典中查找對應的 key,取出對應的值,給模型的屬性賦值。

/** 字典轉模型 **/+ (instancetype)modelWithDict:(NSDictionary *)dict { id objc = [[self alloc] init]; unsigned int count = 0; // 獲取成員屬性數組 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];  // 從字典中取出對應 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);   // 字典轉模型   if (modalClass) {    // 字典轉模型    value = [modalClass modelWithDict:value];   }  }  if ([value isKindOfClass:[NSArray class]]) {   // 判斷對應類有沒有實現字典數組轉模型數組的協議   if ([self respondsToSelector:@selector(arrayContainModelClass)]) {    // 轉換成id類型,就能調用任何對象的方法    id idSelf = self;    // 獲取數組中字典對應的模型    NSString *type = [idSelf arrayContainModelClass][key];    // 生成模型    Class classModel = NSClassFromString(type);    NSMutableArray *arrM = [NSMutableArray array];    // 遍歷字典數組,生成模型數組    for (NSDictionary *dict in value) {     // 字典轉模型     id model = [classModel modelWithDict:dict];     [arrM addObject:model];    }    // 把模型數組賦值給value    value = arrM;   }  }  // KVC 字典轉模型  if (value) {   [objc setValue:value forKey:key];  } } return objc;}

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

總結

以上就是這篇文章的全部內容了,希望本文的內容對大家的學習或者工作具有一定的參考學習價值,如果有疑問大家可以留言交流,謝謝大家對VEVB武林網的支持。


注:相關教程知識閱讀請移步到IOS開發頻道。
發表評論 共有條評論
用戶名: 密碼:
驗證碼: 匿名發表
亚洲香蕉成人av网站在线观看_欧美精品成人91久久久久久久_久久久久久久久久久亚洲_热久久视久久精品18亚洲精品_国产精自产拍久久久久久_亚洲色图国产精品_91精品国产网站_中文字幕欧美日韩精品_国产精品久久久久久亚洲调教_国产精品久久一区_性夜试看影院91社区_97在线观看视频国产_68精品久久久久久欧美_欧美精品在线观看_国产精品一区二区久久精品_欧美老女人bb
26uuu久久噜噜噜噜| 国产亚洲精品久久久久久| 久久这里有精品| 91在线无精精品一区二区| 午夜美女久久久久爽久久| 国产成人激情视频| 91精品在线一区| 日本欧美国产在线| 国产成人精品日本亚洲专区61| 久久久精品2019中文字幕神马| 在线视频一区二区| 奇米4444一区二区三区| 亚洲人成网站色ww在线| 91精品综合视频| 播播国产欧美激情| 亚洲精品www久久久| 岛国av在线不卡| 精品网站999www| 国产精品成av人在线视午夜片| 8090理伦午夜在线电影| 91人人爽人人爽人人精88v| 日韩中文有码在线视频| 国产成人鲁鲁免费视频a| 亚洲精品国产精品自产a区红杏吧| 亚洲欧美激情另类校园| 欧美视频在线观看 亚洲欧| 日韩亚洲国产中文字幕| 国产一区二区日韩| 中文亚洲视频在线| 一区二区三区无码高清视频| 欧美华人在线视频| 5278欧美一区二区三区| 日韩中文字幕第一页| 亚洲精品第一页| 久久久免费精品视频| 国产丝袜一区二区| 欧美尺度大的性做爰视频| 亚洲欧美日韩爽爽影院| 成人黄色片在线| 日韩国产精品亚洲а∨天堂免| 亚洲人av在线影院| 91亚洲午夜在线| 欧美成人三级视频网站| 高清欧美一区二区三区| 国产伦精品一区二区三区精品视频| 国模极品一区二区三区| 欧美电影院免费观看| 欧美乱大交xxxxx另类电影| 国产精品青青在线观看爽香蕉| 久热精品视频在线免费观看| 久久久亚洲精品视频| 日韩视频永久免费观看| 国产精品美乳一区二区免费| 久久久午夜视频| 国产精品av免费在线观看| 91探花福利精品国产自产在线| 岛国av一区二区在线在线观看| 欧美成人中文字幕在线| 亚洲精品v欧美精品v日韩精品| 亚洲精品国产精品久久清纯直播| 久久精视频免费在线久久完整在线看| 在线成人中文字幕| 亚洲成人黄色网| 日韩精品极品在线观看播放免费视频| 性欧美长视频免费观看不卡| 久久久久久久影视| 91久久在线视频| 亚洲精品网址在线观看| 日韩有码视频在线| 中国日韩欧美久久久久久久久| 亚洲成色777777在线观看影院| 亚洲大胆人体视频| 国产精品中文字幕久久久| 不卡在线观看电视剧完整版| 俺去了亚洲欧美日韩| 国产一区二区三区直播精品电影| 亚洲国产91色在线| 国产一区二区三区视频在线观看| 日韩免费在线视频| 日本三级久久久| 国产在线精品成人一区二区三区| 中文字幕欧美在线| 亚洲美女又黄又爽在线观看| 久久香蕉精品香蕉| 日韩精品高清视频| 日韩电影在线观看永久视频免费网站| 久久久久久久久爱| 国产精品视频yy9099| 日韩精品小视频| 最近2019中文字幕mv免费看| 国产精品美女免费视频| 欧美亚洲国产成人精品| 欧美大片免费观看在线观看网站推荐| 伦伦影院午夜日韩欧美限制| 91精品国产综合久久久久久久久| 成人精品视频久久久久| 亚洲人成网在线播放| 日韩高清免费观看| 国产一区红桃视频| 欧美日韩亚洲一区二区三区| 日本三级韩国三级久久| 国内精品久久影院| 久久久精品电影| 国产精品中文字幕在线观看| 亚洲精品国产精品国自产在线| 亚洲曰本av电影| 一个人看的www久久| 97成人精品视频在线观看| 国产男女猛烈无遮挡91| 日韩精品视频中文在线观看| 欧美中文字幕视频在线观看| 欧美日韩国产精品| 久久久久久亚洲精品中文字幕| 亚洲a一级视频| 日韩欧美精品中文字幕| 91天堂在线观看| 国产91在线高潮白浆在线观看| 日韩成人中文字幕在线观看| 欧美一区亚洲一区| 性色av一区二区咪爱| 国产精品午夜一区二区欲梦| 色天天综合狠狠色| 久久综合九色九九| 一个人看的www久久| 2021久久精品国产99国产精品| 午夜精品久久久久久久99热| 欧美黑人性猛交| 久久免费视频网| 日韩av成人在线观看| 精品中文字幕在线| 日韩在线激情视频| 97精品在线视频| 国外视频精品毛片| 国产精品igao视频| 亚洲成人激情小说| 久精品免费视频| 欧美激情网站在线观看| www.精品av.com| 国产日韩中文字幕| 国产盗摄xxxx视频xxx69| 欧美伦理91i| 成人观看高清在线观看免费| 欧美成人在线免费| 在线成人一区二区| 国内伊人久久久久久网站视频| 中文字幕久热精品视频在线| 亚洲第一av网站| 另类天堂视频在线观看| 精品少妇一区二区30p| 亚洲精品国产精品乱码不99按摩| 91精品国产综合久久男男| 日韩av在线影院| 欧美日韩激情小视频| 亚洲www永久成人夜色| 欧美高清无遮挡| 在线电影欧美日韩一区二区私密| 日韩欧美国产高清91| 久久99热这里只有精品国产| 亚洲国产成人一区| 国产精品观看在线亚洲人成网| 国产精品一区二区久久精品| 色综合91久久精品中文字幕| 综合激情国产一区| 美女精品视频一区|