概述

今天我们主要讨论iOS runtime中的一种黑色技术,称为Method Swizzling。字面上理解Method Swizzling可能比较晦涩难懂,毕竟不是中文,不过你可以理解为“移花接木”或者“偷天换日”。

用途

介绍某种技术的用途,最简单的方式就是抛出一些应用场景来引出这种技术的必要性。因此,这里我举个例子如下。

假设工程中有很多ViewController,我需要你统计每个页面间跳转的次数。要求:对原工程的改动越少越好。

针对以上需求,你可能会立马想出以下两种方案:

方案一:

  在每个ViewController的 viewWillAppear 或者 viewDidAppear 方法中对记录跳转次数的某个全局变量(设为 g_viewTransCount )进行计数自增,代码应该是这样的:

- (void)viewDidAppear:(BOOL)animated
{
[super viewDidAppear:animated];
g_viewTransCount++;
}

每个ViewController类中都需要做此操作,显然不合适。因为跳转次数统计这种业务与APP的主业务并没有强关联,上面的代码会造成耦合度过高。随着APP业务的不断扩大,代码中这样的杂质代码会越来越大,维护也越来越困难。而且该方案也违背了我们的要求:对原工程的改动越少越好。因此方案一是个很差的方法。于是我们有了方案二。

方案二:

  有没有某种方法可以不用对每个ViewCotroller都修改呢?有!让每个ViewController都继承某个新的ViewController(设为BaseViewController),然后将统计的代码放到BaseViewCotroller的 viewWillAppear或者viewDidAppear中。这种方案看似较合理,但有以下弊端:

  • 继承自BaseViewCotroller的ViewController中仍旧需要显式调用 [super viewDidAppear:animated];
  • 需要到所有ViewController的头文件中更改其superClass为BaseViewController

可见,方案二虽然相比方案一少一些看得到的“代码杂质”,但对工程的改动同样是巨大的,尤其当工程比较庞大时。

正因为以上方案的不完美,才引出本文的黑科技:Method Swizzling。

先概括一下在上述情景下使用Method Swizzling有哪些优势:

  • 不需要改动现有工程的任何文件
  • 本次统计的代码可复用给其他工程

实现

接下来就是激动人心的Coding Time了。让我们解开Method Swizzling的神秘面纱。直接上代码,有注释。在工程中新建一个UIViewController的category:

#import "UIViewController+swizzling.h"
#import <objc/runtime.h> @implementation UIViewController (swizzling) + (void)load
{
SEL origSel = @selector(viewDidAppear:);
SEL swizSel = @selector(swiz_viewDidAppear:);
[UIViewController swizzleMethods:[self class] originalSelector:origSel swizzledSelector:swizSel];
} //exchange implementation of two methods
+ (void)swizzleMethods:(Class)class originalSelector:(SEL)origSel swizzledSelector:(SEL)swizSel
{
Method origMethod = class_getInstanceMethod(class, origSel);
Method swizMethod = class_getInstanceMethod(class, swizSel); //class_addMethod will fail if original method already exists
BOOL didAddMethod = class_addMethod(class, origSel, method_getImplementation(swizMethod), method_getTypeEncoding(swizMethod));
if (didAddMethod) {
class_replaceMethod(class, swizSel, method_getImplementation(origMethod), method_getTypeEncoding(origMethod));
} else {
//origMethod and swizMethod already exist
method_exchangeImplementations(origMethod, swizMethod);
}
} - (void)swiz_viewDidAppear:(BOOL)animated
{
NSLog(@"I am in - [swiz_viewDidAppear:]");
//handle viewController transistion counting here, before ViewController instance calls its -[viewDidAppear:] method
//需要注入的代码写在此处
[self swiz_viewDidAppear:animated];
} @end

上述代码做了这么一件事:在UIViewController的viewDidAppear:方法调用前插入了跳页计数处理,这一切都在运行时完成。对于上述代码有以下几处需要介绍的:

+ (void)load 方法是一个类方法,当某个类的代码被读到内存后,runtime会给每个类发送 + (void)load 消息。因此 + (void)load 方法是一个调用时机相当早的方法,而且不管父类还是子类,其 + (void)load 方法都会被调用到,很适合用来插入swizzling方法

最核心的代码要数 + (void)swizzleMethods:(Class)class originalSelector:(SEL)origSel swizzledSelector:(SEL)swizSel 了。从函数签名可以看出,该函数是为了交换两个方法内部实现。将目光移到Line23,交换两个方法的内部实现主要依靠两个runtime API:

class_replaceMethod(class, swizSel, method_getImplementation(origMethod), method_getTypeEncoding(origMethod));
method_exchangeImplementations(origMethod, swizMethod);

  

再看一下Line32, - (void)swiz_viewDidAppear:(BOOL)animated 函数看起来像死循环,实际上不会的。原因请看我在下图的注释:

此外,通过断点可以进一步判断出view controller的viewDidAppear实际方法体与category的swiz_viewDidAppear方法的执行先后顺序。为了更直观地说明二者的顺序,我们可以看一下我打出的Log:

通过Log所打印出的顺序足以验证我们的想法。

以上的method swizzling可以应用于iOS的任何类中对其进行代码注入,并且丝毫不影响现有工程的代码。例如,我再举个例子(没办法,我就是喜欢举例子,但我无非是想让你掌握的更多一些)。你想统计整个工程中所有按钮的点击事件的次数,也就是touchUpInside event发生的次数。刚开始你可能会觉得稍微有些没有头绪,因为注入代码的“切入点”相比于UIViewController的viewDidLoad等方法而言不是那么好找。这时候如果你能仔细考虑以下问题或许能找到思路:

  • touchUpInside event发送给什么对象?
  • 该对象本通过什么途径接受这个消息?

第一个问题很好回答,event是发送给UIButton实例,本质上是发送给UIControl实例;

第二个问题你不懂的话就去看看UIControl的头文件找找线索,于是在头文件中我们找到这样一个函数:

- (void)sendAction:(SEL)action to:(id)target forEvent:(UIEvent *)event;

看起来很靠近我们的需求, 事实上的确如此。这要从iOS的事件传递机制说起,当你在iOS设备上触摸一个点时这个触摸动作被包装成一个UIEvent按照UIApplication->UIWindow->UIView的顺序传递下去,当发现最后的接受者是UIControl时就会发送上述消息。因此,我们可以对sendAction:方法进行swizzling代码注入来达到统计按钮点击次数的目的。更深入一些,则需要针对不同的action、target、event的状态进行判断,以达到更精准的统计。关于这一部分内容我将在下一篇iOS动态性系列文章中详细探讨,敬请期待!

OK,文章就到这里,小伙伴们洗洗睡吧。哈哈,开个玩笑,俗话说,“好戏都在后头”,接下来的部分更好用。看来以上的method swizzling代码你是否觉得太复杂了?此外,当你尝试对多个类进行swizzle时会发现很多代码是冗余的,每个category文件的框架都长得差不多。那是否有进一步封装的可能性呢?那是必须的。庆幸的是有团队已经帮我们封装了,我们直接拿来用就可以。这就是有名的Aspect库。

AOP编程以及Aspect库

Aspect库是对面向切面编程(Aspect Oriented Programming)的实现,里面封装了Runtime的方法,也封装了上文的Method Swizzling方法。因此我们也可以看到,Method Swizzling也是AOP编程的一种。Aspect的用途很广泛,这里不具体展开,想了解更多的可以看一下官方github的介绍,已经够详细了。这里我们只介绍其基础应用。Aspect只提供了两个接口:

+ (id<AspectToken>)aspect_hookSelector:(SEL)selector
withOptions:(AspectOptions)options
usingBlock:(id)block
error:(NSError **)error {
return aspect_add((id)self, selector, options, block, error);
} /// @return A token which allows to later deregister the aspect.
- (id<AspectToken>)aspect_hookSelector:(SEL)selector
withOptions:(AspectOptions)options
usingBlock:(id)block
error:(NSError **)error {
return aspect_add(self, selector, options, block, error);
}

使用起来也非常方便,使用Aspect对本文最初提出的需求“统计每个页面间跳转的次数”进行改造,代码变成这样子:

[UIViewController aspect_hookSelector:@selector(viewDidLoad)
withOptions:AspectPositionBefore
usingBlock:^(id<AspectInfo> info){
g_viewTransCount++
NSLog(@"[ASPECT] inject in class instance:%@", [info instance]);
}
error:NULL];

  

将以上代码放到AppDelegate的 didFinishLaunchingWithOptions 函数最开始处即可,你可以参考我在文末贴出的代码,使用一个专门的管理类来管理这些AOP代码。

相比于上半部分的原始Method Swizzling代码,使用Aspect有以下好处:

  • 原则上不需要新建任何文件。这点很好理解,原始Method Swizzling需要新建category文件,当代码注入的需要较多时会出现过多的文件以及冗余代码。
  • 可以对类的实例进行代码注入,因为Aspect提供了实例方法以及类方法

写在最后

Method Swizzling以及Runtime的一些特性就是iOS里的黑科技,如果能灵活应用的话可以在保证解决问题的前提下降低模块之间的耦合度,提高代码的可复用性。至于Method Swizzling与Aspect库的选择因人而异,我个人建议在最初阶段先放下Aspect而只用Method Swizzling原始代码去实现代码注入。掌握本质总是不吃亏的。

本文的示例代码:Github

欢迎关注我的github上的其他代码,别忘记随手点个Star,给我更多支持与鼓励!


原创文章,转载请注明 编程小翁@博客园,邮件zilin_weng@163.com,欢迎各位与我在C/C++/Objective-C/机器视觉等领域展开交流!


【原】iOS动态性(三) Method Swizzling以及AOP编程:在运行时进行代码注入的更多相关文章

  1. Method Swizzling以及AOP编程:在运行时进行代码注入-备用

    概述 今天我们主要讨论iOS runtime中的一种黑色技术,称为Method Swizzling.字面上理解Method Swizzling可能比较晦涩难懂,毕竟不是中文,不过你可以理解为“移花接木 ...

  2. Method Swizzling以及AOP编程:在运行时进行代码注入-b

    概述 今天我们主要讨论iOS runtime中的一种黑色技术,称为Method Swizzling.字面上理解Method Swizzling可能比较晦涩难懂,毕竟不是中文,不过你可以理解为“移花接木 ...

  3. Method Swizzling和AOP(面向切面编程)实践

    Method Swizzling和AOP(面向切面编程)实践 参考: http://www.cocoachina.com/ios/20150120/10959.html 上一篇介绍了 Objectiv ...

  4. IOS 开发之 Method Swizzling + Category

    ios 分类中如果增加的方法与被扩展的类方法名重复,则原方法就没法被调用….看以下例子 例如: @interface ClassA : NSObject - (NSString *) myMethod ...

  5. Method Swizzling 和 AOP 实践(转)

    上一篇介绍了 Objective-C Messaging.利用 Objective-C 的 Runtime 特性,我们可以给语言做扩展,帮助解决项目开发中的一些设计和技术问题.这一篇,我们来探索一些利 ...

  6. IOS 开发之 Method Swizzling

    ios 分类中如果增加的方法与被扩展的类方法名重复,则原方法就没法被调用….看以下例子 例如: @interface ClassA : NSObject - (NSString *) myMethod ...

  7. iOS执行时与method swizzling

    C语言是静态语言,它的工作方式是通过函数调用,这样在编译时我们就已经确定程序怎样执行的.而Objective-C是动态语言,它并不是通过调用类的方法来执行功能,而是给对象发送消息,对象在接收到消息之后 ...

  8. 日志系统实战(二)-AOP动态获取运行时数据

    介绍 这篇距上一篇已经拖3个月之久了,批评自己下. 通过上篇介绍了解如何利用mono反射代码,可以拿出编译好的静态数据.例如方法参数信息之类的. 但实际情况是往往需要的是运行时的数据,就是用户输入等外 ...

  9. Java AOP (2) runtime weaving 【Java 切面编程 (2) 运行时织入】

    接上一篇 Java AOP (1) compile time weaving [Java 切面编程 (1) 编译期织入] Dynamic proxy   动态代理 Befor talking abou ...

随机推荐

  1. Web API接口之FileReader

    Web API接口之FileReader *:first-child { margin-top: 0 !important; } body>*:last-child { margin-botto ...

  2. 使用wireshark抓包分析浏览器无法建立WebSocket连接的问题(server为Alchemy WebSockets组件)

    工作时使用了Websocket技术,在使用的过程中发现,浏览器(Chrome)升级后可能会导致Websocket不可用,更换浏览器后可以正常使用. 近日偶尔一次在本地调试,发现使用相同版本的Chrom ...

  3. JavaScript:声明变量名的语法规则

    一.语法规则 1.变量必须使用字母.下划线(_)或者美元符($)开始. 2.然后可以使用任意多个英文字母.数字.下划线(_)或者美元符($)组成. 3.不能使用JS关键词与保留字. 二.示例 var ...

  4. 用CSS制作带图标的按钮

    先上一张效果图

  5. C#创建安全的字典(Dictionary)存储结构

    在上面介绍过栈(Stack)的存储结构,接下来介绍另一种存储结构字典(Dictionary). 字典(Dictionary)里面的每一个元素都是一个键值对(由二个元素组成:键和值) 键必须是唯一的,而 ...

  6. 百度eCharts体验

    前言 从昨天开始给项目里添加一些图表对比功能,上一个项目里使用的是Highcharts,本打算继续用Highcharts做的,昨天试了下做出来的效果不太好,主要也是因为看的多了没什么新鲜感了,于是便尝 ...

  7. 开源服务专题之------ssh防止暴力破解及fail2ban的使用方法

    15年出现的JAVA反序列化漏洞,另一个是redis配置不当导致机器入侵.只要redis是用root启动的并且未授权的话,就可以通过set方式直接写入一个authorized_keys到系统的/roo ...

  8. 前端读取Excel报表文件

    在实际开发中,经常会遇到导入Excel文件的需求,有的产品人想法更多,想要在前端直接判断文件内容格式是否正确,必填项是否已填写 依据HTML5的FileReader,可以使用新的API打开本地文件(参 ...

  9. ASP.NET Core开发-如何配置Kestrel 网址Urls

    ASP.NET Core中如何配置Kestrel Urls呢,大家可能都知道使用UseUrls() 方法来配置. 今天给介绍全面的ASP.NET Core 配置 Urls,使用多种方式配置Urls. ...

  10. Asp.net 面向接口可扩展框架之应用程序上下文作用域组件

    在团队中推广面向接口开发两年左右,成果总体来说我还是挺满意的,使用面向接口开发的模块使用Unity容器配置的功能非常稳定,便于共享迁移(另一个项目使用只需要复制配置和调用接口即可),如果再配合上DI那 ...