Method Swizzling 与 Aspect Oriented Programming

一些概念:
Method Swizzling:是改变一个selector的实际实现的技术。通过这一技术,我们可以在运行时通过修改类的分发表中selector对应的函数,来修改方法的实现。

Aspect Oriented Programming(面向切面编程):一种编程方式,在指定的地方添加一些自定义代码。
例如一些事务琐碎,跟主要业务逻辑无关,在很多地方都有,又很难抽象出来单独的模块。这种程序设计问题,业界给了一个名字:Cross Cutting Concerns
而用 Method Swizzling 动态给指定的方法添加代码,以解决 Cross Cutting Concerns 的编程方式就叫:Aspect Oriented Programming(面向切面编程)

初识三种method-swizzing的使用

1.method_exchangeImplementations 方法交换 交换SEL->IMP

@interface Change : NSObject
+ (void)eat;
+ (void)run;
@end

@implementation Change
+ (void)load {
    static dispatch_once_t onceToken;
    dispatch_once(&onceToken, ^{
        Method eat = class_getClassMethod(self, @selector(eat));
        Method run = class_getClassMethod(self, @selector(run));
        method_exchangeImplementations(eat, run);
    });
}
+ (void)eat {
    NSLog(@"吃了");
}
+ (void)run {
    NSLog(@"正准备跑");
}
@end

2.class_replaceMethod 取代方法实现 IMP

@interface Replace : NSObject
+ (void)sleep;
@end
@implementation Replace
+ (void)load {
    static dispatch_once_t onceToken;
    dispatch_once(&onceToken, ^{
        class_replaceMethod(self, @selector(sleep), sleepFunction, "");
    });
}
+ (void)sleep {
    NSLog(@"马上睡");
}
void sleepFunction() {
    NSLog(@"还不想睡");
}
@end

3.class_addMethod 添加方法实现 IMP

@interface AddMethod : NSObject
+ (void)work;
@end
@implementation AddMethod
+ (void)load {
    static dispatch_once_t onceToken;
    dispatch_once(&onceToken, ^{
        class_addMethod(objc_getMetaClass(object_getClassName(self)), @selector(work), workFunction, "");
    });
}
void workFunction() {
    NSLog(@"上班");
}
@end

可能出现的错误

1.继承 父类实现了sleep方法 子类没有实现。当进行方法交换的时候 发现将父类的方法实现也交换了

@interface Super : NSObject
+ (void)run;
+ (void)sleep;
+ (void)eat;
@end
@implementation Super
+ (void)run {
    NSLog(@"ErrorSuper_跑了");
}
+ (void)sleep {
    NSLog(@"ErrorSuper_睡了");
}
+ (void)eat {
    NSLog(@"ErrorSuper_吃了");
}
@end
@interface Sub : Super
+ (void)run;
+ (void)sleep;
+ (void)eat;
@end
@implementation Sub
+ (void)load {
    //取代实现
    class_replaceMethod(objc_getMetaClass(object_getClassName(self)), @selector(run), replaceRun, "");
    
    //添加实现
    class_addMethod(objc_getMetaClass(object_getClassName(self)), @selector(eat), addEat, "");
    
    //交换实现
    method_exchangeImplementations(class_getClassMethod(self, @selector(sleep)), class_getClassMethod(self, @selector(changeMethod)));
}


//取代的实现
void replaceRun() {
    NSLog(@"ErrorSub_取代了跑的实现,不想跑");
}

//添加实现
void addEat() {
    NSLog(@"ErrorSub_添加了吃的实现,不想吃");
}


//交换的方法
+ (void)changeMethod {
    NSLog(@"ErrorSub_交换的方法");
}
@end

原因:当进行方法交换的时候,Sub并没有实现sleep方法 那么便会到父类那边寻找sleep的实现。再进行交换的时候,自然交换的便是父类的sleep方法。
解决:给Sub添加上sleep的实现,可以使用class_addMethod动态添加 在进行交换之前添加

2.多次执行方法交换 并没有出现预期效果
关于这一点,只能说是粗心的原因。
假设:方法1与2 进行交换,预期是调用1实现2 调用2实现1
实现:交换的函数执行了两次
分析:
第一次交换,1->2 2->1
第二次交换,1->1 2->2 (在上次的交换中,1的实现是2的实现,2 的实现是1的实现)。
就是换来换去,结果就晕了。
所以,很多资料上面都建议。将方法的交换写在load里面。但是也不能确保不会有人在子类中写上一个[super load] (手贱)。那么在load中再加上一个dispatch_once

比较完整的方法交换形态

@interface Complete : NSObject
+ (void)run;
+ (void)eat;
@end
@implementation Complete
+ (void)load {
    static dispatch_once_t onceToken;
    dispatch_once(&onceToken, ^{
        
        //获取需要交换的方法
        Method run = class_getClassMethod(self, @selector(run));
        Method eat = class_getClassMethod(self, @selector(eat));
        
        //添加eat方法 为什么要添加 是为了防止自身没有 而父类有 进行方法交换的时候 交换了父类的方法实现 显然这不是我们想要的
        //将run的实现添加到 新添加的eat方法上
        BOOL isAdd = class_addMethod(objc_getMetaClass(object_getClassName(self)), @selector(eat), method_getImplementation(run), method_getTypeEncoding(run));
        
        //判断
        if (isAdd) {
            //添加成功 表示自身确实没有eat方法
            //那么这个时候只需要将eat的方法实现 替换到 run方法上即可
            //是否还记得 判断之前我们添加过SEL为eat  IMP为run的实现 的eat方法
            class_replaceMethod(self, @selector(run), method_getImplementation(eat), method_getTypeEncoding(eat));
        }else {
            //添加失败  表示自身存在eat方法
            //那么这个时候只需要将 run 与 eat 的IMP(方法实现) 进行交换就好了
            method_exchangeImplementations(run, eat);
        }
    });
}

+ (void)run {
    NSLog(@"跑了");
}
+ (void)eat {
    NSLog(@"吃了");
}
@end

小例子 利用类别判断当前是那个控制器 同时也可以做很多事 例如统计

@implementation UIViewController (swizzing)
+ (void)load {
    static dispatch_once_t onceToken;
    dispatch_once(&onceToken, ^{
        Method originalMethod = class_getInstanceMethod(self, @selector(viewWillAppear:));
        Method swizzMethod = class_getInstanceMethod(self, @selector(swizz_viewWillAppear:));
        
        BOOL isAdd = class_addMethod(self, @selector(viewWillAppear:), method_getImplementation(swizzMethod), method_getTypeEncoding(swizzMethod));
        
        if (isAdd) {
            class_replaceMethod(self, @selector(swizz_viewWillAppear:), method_getImplementation(originalMethod), method_getTypeEncoding(originalMethod));
        }else {
            method_exchangeImplementations(originalMethod, swizzMethod);
        }
    });
}

- (void)swizz_viewWillAppear:(BOOL)animated {
    //这里调用自身并不会引起递归
    //记得两个方法的实现已经进行交换了么
    //这里的调用 是调用了系统的方法
    [self swizz_viewWillAppear:animated];
    if ([self isKindOfClass:[ViewController class]]) {
        NSLog(@"当前控制器是ViewController");
    }else if ([self isKindOfClass:[PVC1 class]]) {
        NSLog(@"当前控制器是PVC1");
    }else if ([self isKindOfClass:[PVC2 class]]) {
        NSLog(@"当前控制器是PVC2");
    }else if ([self isKindOfClass:[PVC3 class]]) {
        NSLog(@"当前控制器是PVC3");
    }
}
@end

当然了,实现这种事情,有很多种方法。
但是同时也有很多局限性,可以参考知识链接里面的解释;
在类别中 你没法 super .... 例如[super viewWillAppear]

Logging 的代码都很相似,通过继承或类别重写相关方法是可以把它从主要逻辑中剥离出来。但同时也带来新的问题:

1.你需要继承 UIViewController, UITableViewController, UICollectionViewController 所有这些 ViewController ,或者给他们添加类别;

2.每个 ViewController 里的 ButtonClick 方法命名不可能都一样;

3.你不能控制别人如何去实例化你的子类;

4.对于类别,你没办法调用到原来的方法实现。大多时候,我们重写一个方法只是为了添加一些代码,而不是完全取代它。

5.如果有两个类别都实现了相同的方法,运行时没法保证哪一个类别的方法会给调用。

关于AOP,参考知识链接里面的东西。能够了解的更加透彻
知识链接:
AOP
SEL-IMP
Aspects

体验小Demo:demo

分享一个关于main函数调用之前系统所做的一些事情:
https://www.jianshu.com/p/43db6b0aab8e
https://blog.csdn.net/yu_4074/article/details/54966782

最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 204,732评论 6 478
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 87,496评论 2 381
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 151,264评论 0 338
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 54,807评论 1 277
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 63,806评论 5 368
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 48,675评论 1 281
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 38,029评论 3 399
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 36,683评论 0 258
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 41,704评论 1 299
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 35,666评论 2 321
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 37,773评论 1 332
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 33,413评论 4 321
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 39,016评论 3 307
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 29,978评论 0 19
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 31,204评论 1 260
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 45,083评论 2 350
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 42,503评论 2 343

推荐阅读更多精彩内容

  • 转至元数据结尾创建: 董潇伟,最新修改于: 十二月 23, 2016 转至元数据起始第一章:isa和Class一....
    40c0490e5268阅读 1,678评论 0 9
  • 继上Runtime梳理(四) 通过前面的学习,我们了解到Objective-C的动态特性:Objective-C不...
    小名一峰阅读 741评论 0 3
  • 转载:http://www.cocoachina.com/ios/20161102/17920.html 因为Ob...
    F麦子阅读 664评论 0 1
  • Spring Cloud为开发人员提供了快速构建分布式系统中一些常见模式的工具(例如配置管理,服务发现,断路器,智...
    卡卡罗2017阅读 134,596评论 18 139
  • 我们有时会很困惑,到底是什么原因促使我们跟眼前这个人走入婚姻的? 很多人会问对方爱不爱自己,认为这个问题是结婚前最...
    罗掌柜real阅读 320评论 0 0