亚洲乱码中文字幕综合,中国熟女仑乱hd,亚洲精品乱拍国产一区二区三区,一本大道卡一卡二卡三乱码全集资源,又粗又黄又硬又爽的免费视频

iOS中Runtime的幾種基本用法記錄

 更新時間:2018年07月06日 15:50:11   作者:isabadguy  
RunTime顧名思義運行時,就是系統(tǒng)在運行的時候的一些機制,最主要的是消息機制。下面這篇文章主要給大家介紹了關(guān)于iOS中Runtime的幾種基本用法,文中通過示例代碼介紹的非常詳細,需要的朋友下面隨著小編來一起學習學習吧

Runtime 介紹

這不是一遍介紹關(guān)于Runtime實現(xiàn)細節(jié)的文章,而是怎么利用Objective-C提供的Runtime API進行開發(fā)的文章!

Objective-C擁有相當多的動態(tài)特性,這些特性在運行程序時候發(fā)揮作用.

Objctive-C Runtime是個運行時的庫,由C和匯編實現(xiàn)。通過Runtime封裝的C結(jié)構(gòu)體和函數(shù)可以在程序運行時創(chuàng)建、檢查和修改類以及對象及其方法,甚至可以替換或交換方法的實現(xiàn)。

下面記錄一下關(guān)于Runtime的一些基本用法

1)消息機制

在OOP術(shù)語中,消息傳遞是指一種在對象之間發(fā)送和接收消息的通信模式。

在Objective-C中,消息傳遞用于在調(diào)用類和類實例的方法,即接收者接收需要執(zhí)行的消息。

使用案例

// 通過類名獲取類
Class catClass = objc_getClass("Cat"); 
 
//注意Class實際上也是對象,所以同樣能夠接受消息,向Class發(fā)送alloc消息
Cat *cat = objc_msgSend(catClass, @selector(alloc)); 
 
//發(fā)送init消息給Cat實例cat
cat = objc_msgSend(cat, @selector(init)); 
 
//發(fā)送eat消息給cat,即調(diào)用eat方法
objc_msgSend(cat, @selector(eat));
 
//匯總消息傳遞過程
objc_msgSend(objc_msgSend(objc_msgSend(objc_getClass("Cat"), sel_registerName("alloc")), sel_registerName("init")), sel_registerName("eat"));

2)方法交換 Method Swizzling

Objective-C 提供了一下API用于動態(tài)替換類方法或者實例方法的實現(xiàn):

  • class_replaceMethod 替換類方法的定義
  • method_exchangeImplementations 交換兩個方法的實現(xiàn)(具體使用案例如下)
  • method_setImplementation 設(shè)置一個方法的實現(xiàn)

注:class_replaceMethod 試圖替換一個不存在的方法時候,會調(diào)用class_addMethod為該類增加一個新方法

使用案例

//Cat.m

+ (void)load{
 Method eatMethod = class_getInstanceMethod(self, @selector(eat));
  Method shirtMethod = class_getInstanceMethod(self, @selector(shirt));
 
 method_exchangeImplementations(eatMethod, shirtMethod);
}

- (void)eat{
 NSLog(@"cat eat....");
}

- (void)shirt{
 NSLog(@"cat shirt....");
}

3)動態(tài)加載方法

當調(diào)用一個未實現(xiàn)的方法,或者說發(fā)送未知的消息給接收者時候,消息的接受者會調(diào)用resolveInstanceMethod

使用案例

// Cat.m

//An Objective-C method is simply a C function that take at least two arguments—self and _cmd. 
void run(id self, SEL _cmd, NSNumber *number){
 NSLog(@"run for %@", number);
}

//收到run:消息時候,為該類添加一個方法實現(xiàn)
+ (BOOL)resolveInstanceMethod:(SEL)sel{
 if(sel == NSSelectorFromString(@"run:")){
  class_addMethod(self, @selector(run:), run, "v@:@");
  return YES;
 }
 return [super resolveInstanceMethod:sel];
}

//另外針對類方法的為 resolveClassMethod

4)消息轉(zhuǎn)發(fā)

// 第一步,消息接收者沒有找到對應的方法時候,會先調(diào)用此方法,可在此方法實現(xiàn)中動態(tài)添加新的方法
// 返回YES表示相應selector的實現(xiàn)已經(jīng)被找到,或者添加新方法到了類中,否則返回NO
+ (BOOL)resolveInstanceMethod:(SEL)sel {
 return YES;
}

// 第二步, 如果第一步的返回NO或者直接返回了YES而沒有添加方法,該方法被調(diào)用
// 在這個方法中,我們可以指定一個可以返回一個可以響應該方法的對象, 注意如果返回self就會死循環(huán)
- (id)forwardingTargetForSelector:(SEL)aSelector {
 return nil;
}


// 第三步, 如果forwardingTargetForSelector:返回了nil,則該方法會被調(diào)用,系統(tǒng)會詢問我們要一個合法的『類型編碼(Type Encoding)』
// 若返回 nil,則不會進入下一步,而是無法處理消息
- (NSMethodSignature *)methodSignatureForSelector:(SEL)aSelector {
 return [NSMethodSignature signatureWithObjCTypes:"v@:"];
}

// 當實現(xiàn)了此方法后,-doesNotRecognizeSelector: 將不會被調(diào)用
// 在這里進行消息轉(zhuǎn)發(fā)
- (void)forwardInvocation:(NSInvocation *)anInvocation {
 // 在這里可以改變方法選擇器
 [anInvocation setSelector:@selector(unknown)];
 // 改變方法選擇器后,需要指定消息的接收者
 [anInvocation invokeWithTarget:self];
}

- (void)unknown {
 NSLog(@"unkown method.......");
}

// 如果沒有實現(xiàn)消息轉(zhuǎn)發(fā) forwardInvocation 則調(diào)用此方法
- (void)doesNotRecognizeSelector:(SEL)aSelector {
 NSLog(@"unresolved method :%@", NSStringFromSelector(aSelector));
}

注: 『類型編碼(Type Encoding)』

5)動態(tài)關(guān)聯(lián)屬性

對象在內(nèi)存中的排布可以看成一個結(jié)構(gòu)體,該結(jié)構(gòu)體的大小并不能動態(tài)變化,所以無法在運行時動態(tài)給對象增加成員變量。相對的,對象的方法定義都保存在類的可變區(qū)域中。

如下圖所示為Class 的描述信息,其中methodList為可訪問類中定義的方法的指針的指針,通過修改該指針所指向的指針的值,我們可以實現(xiàn)為類動態(tài)增加方法實現(xiàn)。

因此,我們可以實現(xiàn)動態(tài)為一個類增加成員方法,但是卻不能直接為類增加成員變量,這就是category的實現(xiàn)原理。

//<objc/runtime.h>

struct objc_class {
 Class isa OBJC_ISA_AVAILABILITY;

#if !__OBJC2__
 Class super_class          OBJC2_UNAVAILABLE;
 const char *name           OBJC2_UNAVAILABLE;
 long version            OBJC2_UNAVAILABLE;
 long info            OBJC2_UNAVAILABLE;
 long instance_size          OBJC2_UNAVAILABLE;
 struct objc_ivar_list *ivars        OBJC2_UNAVAILABLE;
 struct objc_method_list **methodLists     OBJC2_UNAVAILABLE;
 struct objc_cache *cache         OBJC2_UNAVAILABLE;
 struct objc_protocol_list *protocols      OBJC2_UNAVAILABLE;
#endif

} OBJC2_UNAVAILABLE;

使用案例

//Cat+Extend.h

@interface Cat (extend)

@property(nonatomic, copy) NSString *name;

@end


//Cat+Extend.m

@implementation Cat (extend)

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

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

@end

6)字典轉(zhuǎn)模型應用

通過Class的結(jié)構(gòu)體內(nèi)容,可以看到ivars指針指向包含了類中成員變量的結(jié)構(gòu)體,通過它可以得到類中定義的成員變量,而Objective-C中提供了相應的API方法: class_copyIvarList

//<objc/runtime.h>

struct objc_class {
 Class isa OBJC_ISA_AVAILABILITY;

#if !__OBJC2__
 Class super_class          OBJC2_UNAVAILABLE;
 const char *name           OBJC2_UNAVAILABLE;
 long version            OBJC2_UNAVAILABLE;
 long info            OBJC2_UNAVAILABLE;
 long instance_size          OBJC2_UNAVAILABLE;
 struct objc_ivar_list *ivars        OBJC2_UNAVAILABLE;
 struct objc_method_list **methodLists     OBJC2_UNAVAILABLE;
 struct objc_cache *cache         OBJC2_UNAVAILABLE;
 struct objc_protocol_list *protocols      OBJC2_UNAVAILABLE;
#endif

} OBJC2_UNAVAILABLE;

使用案例

//Cat.h

@property(nonatomic, copy) NSString *cid;

@property(nonatomic, copy) NSString *age;

+ (instancetype)modelWithDict:(NSDictionary *)dict;


//Cat.m

+ (instancetype)modelWithDict:(NSDictionary *)dict{
 id model = [[self alloc] init];
 unsigned int count = 0;
 
 Ivar *ivars = class_copyIvarList(self, &count);
 for (int i = 0 ; i < count; i++) {
  Ivar ivar = ivars[i];
  
  NSString *ivarName = [NSString stringWithUTF8String:ivar_getName(ivar)];
  
  //這里注意,拿到的成員變量名為_cid,_age
  ivarName = [ivarName substringFromIndex:1];
  id value = dict[ivarName];
  
  [model setValue:value forKeyPath:ivarName];
 }
 
 return model;
}

總結(jié)

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

相關(guān)文章

  • iOS開發(fā)之UIScrollView控件詳解

    iOS開發(fā)之UIScrollView控件詳解

    UIScrollView是一個非常重要的控件,其可以展示比設(shè)備屏幕更大區(qū)域的內(nèi)容,我們可以通過手指滑動來查看內(nèi)容視圖的每一部分內(nèi)容,也可以通過手指捏合來對內(nèi)容視圖進行縮放操作,我們每天開發(fā)中都不斷顯式或隱式地與UIScrollView打交道,下面給大家詳細介紹UIScrollView控件。
    2016-09-09
  • IOS 開發(fā)之對象為空的判斷(nil、null)詳解

    IOS 開發(fā)之對象為空的判斷(nil、null)詳解

    這篇文章主要介紹了IOS 開發(fā)之對象為空的判斷(nil、null)詳解的相關(guān)資料,需要的朋友可以參考下
    2017-02-02
  • iOS坐標系的深入探究

    iOS坐標系的深入探究

    這篇文章主要給大家介紹了關(guān)于iOS坐標系的相關(guān)資料,文中通過示例代碼介紹的非常詳細,對大家的學習或者工作具有一定的參考學習價值,需要的朋友們下面隨著小編來一起學習學習吧
    2018-11-11
  • iOS中設(shè)置圓角的幾種方法示例

    iOS中設(shè)置圓角的幾種方法示例

    這篇文章主要介紹了iOS中設(shè)置圓角的三種方法,其中包括使用layer屬性、使用繪圖設(shè)置圓角以及通過另一張mask圖創(chuàng)建新圖,需要的朋友可以參考借鑒,下面來一起看看吧。
    2017-03-03
  • iOS 二維碼掃描相關(guān)功能實現(xiàn)

    iOS 二維碼掃描相關(guān)功能實現(xiàn)

    這篇文章主要介紹了iOS 二維碼掃描相關(guān)功能實現(xiàn),小編覺得挺不錯的,現(xiàn)在分享給大家,也給大家做個參考。一起跟隨小編過來看看吧
    2018-09-09
  • iOS基礎(chǔ)知識之@property 和 Ivar 的區(qū)別

    iOS基礎(chǔ)知識之@property 和 Ivar 的區(qū)別

    這篇文章主要介紹了iOS基礎(chǔ)知識之@property 和 Ivar 的區(qū)別介紹,非常不錯,具有參考借鑒價值,需要的朋友可以參考下
    2016-08-08
  • iOS App開發(fā)中的UIPageControl分頁控件使用小結(jié)

    iOS App開發(fā)中的UIPageControl分頁控件使用小結(jié)

    UIPageControl分頁控件的例子簡單來說即是我們平時翻動多個桌面頁時及底部帶有的圓點頁碼標注,這里我們來看一下iOS App開發(fā)中的UIPageControl分頁控件使用小結(jié),需要的朋友可以參考下
    2016-06-06
  • iOS 加載Bundle文件的實例代碼

    iOS 加載Bundle文件的實例代碼

    這篇文章主要介紹了iOS 加載Bundle文件的實例代碼,代碼簡單易懂,非常不錯,需要的朋友參考下
    2016-12-12
  • IOS ObjectC與javascript交互詳解及實現(xiàn)代碼

    IOS ObjectC與javascript交互詳解及實現(xiàn)代碼

    這篇文章主要介紹了IOS OC與js交互詳解及實現(xiàn)代碼的相關(guān)資料,需要的朋友可以參考下
    2017-03-03
  • iOS AFNetworking各種功能封裝類代碼

    iOS AFNetworking各種功能封裝類代碼

    下面小編就為大家分享一篇iOS AFNetworking各種功能封裝類代碼,具有很好的參考價值,希望對大家有所幫助。一起跟隨小編過來看看吧
    2018-01-01

最新評論