Runtime動態(tài)添加方法

// 2019開工第一天座慰,今年想好好學(xué)習(xí);看了下Runtime鹃祖,然后簡單記錄下钓猬,怕自己忘記了:

// 添加方法
BOOL class_addMethod ( Class cls, SEL name, IMP imp, const char *types );
// 獲取實(shí)例方法
Method class_getInstanceMethod ( Class cls, SEL name );
// 獲取類方法
Method class_getClassMethod ( Class cls, SEL name );
// 獲取所有方法的List
Method * class_copyMethodList ( Class cls, unsigned int *outCount );
// 替代方法的實(shí)現(xiàn)
IMP class_replaceMethod ( Class cls, SEL name, IMP imp, const char *types );
// 返回方法的具體實(shí)現(xiàn)
IMP class_getMethodImplementation ( Class cls, SEL name );
IMP class_getMethodImplementation_stret ( Class cls, SEL name );
// 類實(shí)例是否響應(yīng)指定的selector
BOOL class_respondsToSelector ( Class cls, SEL sel );
??:當(dāng)判斷一個實(shí)例方法是否實(shí)現(xiàn)時,第一個參數(shù)要用類對象匀借,也就是[Person class]颜阐。
當(dāng)判斷一個類方法是否實(shí)現(xiàn)時,第一個參數(shù)要傳元類吓肋,也就是object_getClass([Person class])凳怨。

objc_msgSend 動態(tài)添加方法

申明一個ViewController類和Person類

在person類中實(shí)現(xiàn)方法

/**
 無參數(shù)無返回值
 */
-(void)fun
{
    NSLog(@"fun");
}

/**
 有參數(shù)無返回值

 @param food food
 @param some some
 */
-(void)eat:(NSString *)food say:(NSString *)some
{
    NSLog(@"%@ %@",food, some);
    
}

/**
 有參數(shù)有返回值

 @param value value
 @return NSString
 */
- (NSString *)getString:(NSString *)value{
    
    NSLog(@"ba\\value = %@",value);
    
    return value;
}

然后在VC中對應(yīng)實(shí)現(xiàn)相應(yīng)的方法:

    //  無參數(shù)無返回值
    ((void (*)(id, SEL))objc_msgSend)(p, @selector(fun)); //正確寫法
  
    //  有參數(shù)無返回值
    void (*glt_msgsend)(id, SEL, NSString *, NSString *) = (void (*)(id, SEL, NSString *, NSString *))objc_msgSend;
    glt_msgsend(p, @selector(eat:say:), @"123", @"456");
  
   //  有參數(shù)有返回值
    NSString* (*String_msgsend)(id, SEL, NSString  *) = (NSString * (*)(id, SEL, NSString *))objc_msgSend; 
    String_msgsend(p, @selector(getString:), @"value");

class_addMethod 使用

   /*
     1.cls 哪個類
     2.SEL
     3.IMP
     4.返回值類型!
     */
class_addMethod ( Class cls, SEL name, IMP imp, const char *types );

如果知道對應(yīng)的函數(shù)返回值和類型,則可以直接使用

// i:返回值類型int是鬼,若是v則表示void

class_addMethod(self, sel, (IMP)haha, "v@:@");

void haha(id obj,SEL sel,NSString * objc){
    NSLog(@"%@--%@--%@",obj,NSStringFromSelector(sel),objc);
}

否則可以用下面的方法:

  SEL swizzledMethod = @selector(systemMethod_PrintLog);
  Method swizzledMethod = class_getInstanceMethod(class, originalSelector);
  class_addMethod(class,
                        originalSelector,
                        method_getImplementation(swizzledMethod),
                        method_getTypeEncoding(swizzledMethod));

附加:方法的替換

load是只要類所在文件被引用就會被調(diào)用肤舞,而initialize是在類或者其子類的第一個方法被調(diào)用前調(diào)用。所以如果類沒有被引用進(jìn)項(xiàng)目均蜜,就不會有l(wèi)oad調(diào)用李剖;但即使類文件被引用進(jìn)來,但是沒有使用囤耳,那么initialize也不會被調(diào)用篙顺。load的調(diào)用會引起initialize的調(diào)用。

/**
 消息替換方法
 */
+ (void)load {
    NSLog(@"load");
    
    static dispatch_once_t onceToken;
    dispatch_once(&onceToken, ^{
        Class class = [self class];
        
        SEL originalSelector = @selector(systemMethod_PrintLog);
        SEL swizzledSelector = @selector(ll_imageName);
        
        Method originalMethod = class_getInstanceMethod(class, originalSelector);
        Method swizzledMethod = class_getInstanceMethod(class, swizzledSelector);
        
        BOOL didAddMethod =
        class_addMethod(class,
                        originalSelector,
                        method_getImplementation(swizzledMethod),
                        method_getTypeEncoding(swizzledMethod));
        
        if (didAddMethod) {
            class_replaceMethod(class,
                                swizzledSelector,
                                method_getImplementation(originalMethod),
                                method_getTypeEncoding(originalMethod));
        } else {
            
            method_exchangeImplementations(originalMethod, swizzledMethod);
        }
    });
    
}

+(void)initialize{
    
     NSLog(@"initialize");
    
}

當(dāng)一個methad沒實(shí)現(xiàn)可能會引起Crash時候充择,方法的攔截

//是為類方法進(jìn)行決議德玫。class_addMethod添加Method
+ (BOOL)resolveClassMethod:(SEL)name;  
// 是為對象方法進(jìn)行決議,
+ (BOOL)resolveInstanceMethod:(SEL)name;  

eg1:

//如果該類接收到一個沒有實(shí)現(xiàn)的實(shí)例方法,就會來到這里

+(BOOL)resolveInstanceMethod:(SEL)sel
{
//    NSLog(@"%@",NSStringFromSelector(sel));
    //動態(tài)添加一個方法!!
    /*
     1.cls 哪個類
     2.SEL
     3.IMP
     4.返回值類型!
     */
    class_addMethod(self, sel, (IMP)haha, "v@:@");
    
    return [super resolveInstanceMethod:sel];
}

void haha(id obj,SEL sel,NSString * objc){
    NSLog(@"%@--%@--%@",obj,NSStringFromSelector(sel),objc);
}

eg2:
- (void) run {
    NSLog(@"%s", __func__);
}

+ (BOOL)resolveInstanceMethod:(SEL)sel {
    NSLog(@"%s", __func__);
    
//    // 元類
//    // 實(shí)例對象椎麦、類對象宰僧、元類對象
//    if (sel == @selector(walk)) {
//        return class_addMethod(self, sel, (IMP)walk, "v@:");
//    }
    
    if (sel == @selector(walk)) {
        Method runMethod = class_getInstanceMethod(self, @selector(run));
        IMP runIMP = method_getImplementation(runMethod);
        const char* types = method_getTypeEncoding(runMethod);
        NSLog(@"%s", types);
        return class_addMethod(self, sel, runIMP, types);
    }
    
    return [super resolveInstanceMethod:sel];
}

eg3:
+ (void) run {
    NSLog(@"%s", __func__);
}

+ (BOOL)resolveClassMethod:(SEL)sel {
    
    if (sel == @selector(walk)) {
        Method runMethod = class_getClassMethod(self, @selector(run));
        IMP runIMP = method_getImplementation(runMethod);
        const char* types = method_getTypeEncoding(runMethod);
        NSLog(@"%s", types);
        return class_addMethod(object_getClass(self), sel, runIMP, types);
    }
    return [super resolveClassMethod:sel];
}

參考鏈接:
Runtime基礎(chǔ)使用場景-攔截替換方法
class_addMethod以及消息轉(zhuǎn)發(fā)機(jī)制
objc_msgSend的使用注意事項(xiàng)

最后編輯于
?著作權(quán)歸作者所有,轉(zhuǎn)載或內(nèi)容合作請聯(lián)系作者
  • 序言:七十年代末,一起剝皮案震驚了整個濱河市观挎,隨后出現(xiàn)的幾起案子琴儿,更是在濱河造成了極大的恐慌段化,老刑警劉巖,帶你破解...
    沈念sama閱讀 218,451評論 6 506
  • 序言:濱河連續(xù)發(fā)生了三起死亡事件造成,死亡現(xiàn)場離奇詭異显熏,居然都是意外死亡,警方通過查閱死者的電腦和手機(jī)谜疤,發(fā)現(xiàn)死者居然都...
    沈念sama閱讀 93,172評論 3 394
  • 文/潘曉璐 我一進(jìn)店門佃延,熙熙樓的掌柜王于貴愁眉苦臉地迎上來,“玉大人夷磕,你說我怎么就攤上這事∽醒兀” “怎么了坐桩?”我有些...
    開封第一講書人閱讀 164,782評論 0 354
  • 文/不壞的土叔 我叫張陵,是天一觀的道長封锉。 經(jīng)常有香客問我绵跷,道長,這世上最難降的妖魔是什么成福? 我笑而不...
    開封第一講書人閱讀 58,709評論 1 294
  • 正文 為了忘掉前任碾局,我火速辦了婚禮,結(jié)果婚禮上奴艾,老公的妹妹穿的比我還像新娘净当。我一直安慰自己,他們只是感情好蕴潦,可當(dāng)我...
    茶點(diǎn)故事閱讀 67,733評論 6 392
  • 文/花漫 我一把揭開白布像啼。 她就那樣靜靜地躺著,像睡著了一般潭苞。 火紅的嫁衣襯著肌膚如雪忽冻。 梳的紋絲不亂的頭發(fā)上,一...
    開封第一講書人閱讀 51,578評論 1 305
  • 那天此疹,我揣著相機(jī)與錄音僧诚,去河邊找鬼。 笑死蝗碎,一個胖子當(dāng)著我的面吹牛湖笨,可吹牛的內(nèi)容都是我干的。 我是一名探鬼主播衍菱,決...
    沈念sama閱讀 40,320評論 3 418
  • 文/蒼蘭香墨 我猛地睜開眼赶么,長吁一口氣:“原來是場噩夢啊……” “哼!你這毒婦竟也來了脊串?” 一聲冷哼從身側(cè)響起辫呻,我...
    開封第一講書人閱讀 39,241評論 0 276
  • 序言:老撾萬榮一對情侶失蹤清钥,失蹤者是張志新(化名)和其女友劉穎,沒想到半個月后放闺,有當(dāng)?shù)厝嗽跇淞掷锇l(fā)現(xiàn)了一具尸體祟昭,經(jīng)...
    沈念sama閱讀 45,686評論 1 314
  • 正文 獨(dú)居荒郊野嶺守林人離奇死亡,尸身上長有42處帶血的膿包…… 初始之章·張勛 以下內(nèi)容為張勛視角 年9月15日...
    茶點(diǎn)故事閱讀 37,878評論 3 336
  • 正文 我和宋清朗相戀三年怖侦,在試婚紗的時候發(fā)現(xiàn)自己被綠了篡悟。 大學(xué)時的朋友給我發(fā)了我未婚夫和他白月光在一起吃飯的照片。...
    茶點(diǎn)故事閱讀 39,992評論 1 348
  • 序言:一個原本活蹦亂跳的男人離奇死亡匾寝,死狀恐怖搬葬,靈堂內(nèi)的尸體忽然破棺而出,到底是詐尸還是另有隱情艳悔,我是刑警寧澤急凰,帶...
    沈念sama閱讀 35,715評論 5 346
  • 正文 年R本政府宣布,位于F島的核電站猜年,受9級特大地震影響抡锈,放射性物質(zhì)發(fā)生泄漏。R本人自食惡果不足惜乔外,卻給世界環(huán)境...
    茶點(diǎn)故事閱讀 41,336評論 3 330
  • 文/蒙蒙 一床三、第九天 我趴在偏房一處隱蔽的房頂上張望。 院中可真熱鬧杨幼,春花似錦撇簿、人聲如沸。這莊子的主人今日做“春日...
    開封第一講書人閱讀 31,912評論 0 22
  • 文/蒼蘭香墨 我抬頭看了看天上的太陽。三九已至歹撒,卻和暖如春莲组,著一層夾襖步出監(jiān)牢的瞬間,已是汗流浹背暖夭。 一陣腳步聲響...
    開封第一講書人閱讀 33,040評論 1 270
  • 我被黑心中介騙來泰國打工锹杈, 沒想到剛下飛機(jī)就差點(diǎn)兒被人妖公主榨干…… 1. 我叫王不留,地道東北人迈着。 一個月前我還...
    沈念sama閱讀 48,173評論 3 370
  • 正文 我出身青樓竭望,卻偏偏與公主長得像,于是被迫代替她去往敵國和親裕菠。 傳聞我的和親對象是個殘疾皇子咬清,可洞房花燭夜當(dāng)晚...
    茶點(diǎn)故事閱讀 44,947評論 2 355

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

  • 文中的實(shí)驗(yàn)代碼我放在了這個項(xiàng)目中。 以下內(nèi)容是我通過整理[這篇博客] (http://yulingtianxia....
    茗涙閱讀 923評論 0 6
  • 本文詳細(xì)整理了 Cocoa 的 Runtime 系統(tǒng)的知識,它使得 Objective-C 如虎添翼旧烧,具備了靈活的...
    lylaut閱讀 800評論 0 4
  • 本文轉(zhuǎn)載自:http://yulingtianxia.com/blog/2014/11/05/objective-...
    ant_flex閱讀 761評論 0 1
  • 寫在前面:關(guān)于SEL影钉、Method、IMP之前的關(guān)系掘剪,請參考CornBallast寫的Runtime奇技淫巧之類(...
    Aaron升閱讀 488評論 0 1
  • 我們常常會聽說 Objective-C 是一門動態(tài)語言平委,那么這個「動態(tài)」表現(xiàn)在哪呢?我想最主要的表現(xiàn)就是 Obje...
    Ethan_Struggle閱讀 2,195評論 0 7