(方法调配)Method Swizzling
一、概念
方法调配:因为Objective-C是运行时语言,也就是说究竟会调用何种方法要在运行期才能解析出来。那么我们其实也可以在运行时改变选择子名称。这样我们既不需要查看到源代码,又没有必要去重写子类来覆写方法就能改变类本身的功能。这样一来新功能就会在类的所有实例中表现出来,而不仅限于那些重写子类的实例。这种方案就叫做“方法调配”(method swizzling)。
IMP:类方法列表会把选择子的名称映射到方法的实现上,使得动态消息派发系统能够据次找到应该调用的方法,这些方法均以指针的形式来表示,这种指针叫做IMP。其原型如下:
id(* IMP)(id,SEL,.....)
SEL:选择器用于表示一个方法在运行时的名字,一个方法的选择器是一个注册到(或映射到)Objective-C运行时中的C字符串,它是由编译器生成并在类加载的时候被运行时系统自动映射。
一个类(Class)维护一张调度表(dispatch table)用于解析运行时发送的消息;调度表中的每个实体(entry)都是一个方法(Method),其中key值是一个唯一的名字——选择器(SEL),它对应到一个实现(IMP)——实际上就是指向标准C函数的指针。
二、方法调配的实现
首先看两张图片:
![](http://images2015.cnblogs.com/blog/527522/201509/527522-20150908102955809-114617068.png)
![](http://images2015.cnblogs.com/blog/527522/201509/527522-20150908103004450-1136892141.png)
以上图2-3是NSString可以相应的选择子lowercaseString,uppercaseString,capitalizedString。每个选择子都映射到了不同的IMP。现在我们通过运行系统提供的方法来将选择子映射表改为2-4所示的映射表。
先简单说一下:我们没有重写子类或者其他,而是修改了方法的布局,这样就会反映到所有的NSString实例上。以下是一些基本的函数操作以及作用:
①、交换方法的实现
void method_exchangeImplementations(Method m1,Method m2)
此方法的参数是两个方法实现,方法实现可以通过以下函数获得:
Method class_getInstanceMethod(Class class,SEL aSelector)
次方法根据给定的选择从类中取出相应的方法。现在举个例子交换lowcaseString和uppercaseString:
Method originMethod = class_getInstanceMethod([NSString class],@selector(lowcaseString)); Method swipMethod = class_getInstanceMethod([NSString class],@selector(uppercaseString)); method_exchangeImplementations(originMethod,swipMethod);
此时如果我们调用NSString的lowcaseString那么实现的结果是uppercaseString的实现。(即如果调用小写转换则是大写转换)。
直接交换的意义不大,我们一般都会用在想重写一个类的方法,然后添加自己的东西,然后将这个方法和类现有的方法进行交换。
②、添加新的方法:
![复制代码](http://common.cnblogs.com/images/copycode.gif)
+ (IMP)swizzleSelector:(SEL)origSelector withIMP:(IMP)newIMP { Class class = [self class]; Method origMethod = class_getInstanceMethod(class, origSelector); IMP origIMP = method_getImplementation(origMethod); if(!class_addMethod(self, origSelector, newIMP, method_getTypeEncoding(origMethod))) method_setImplementation(origMethod, newIMP); return origIMP; }
![复制代码](http://common.cnblogs.com/images/copycode.gif)
具体就不在介绍了。你懂的,,,,
三、总结:
1、在运行期间可以通过方法调配新增或者替换选择子对应方法的实现。
2、使用另一份实现来替换原有的实现,这道程序就叫做方法调配。
3、一定要注意使用,一般来说只有在调试程序的时候才去进行方法调配,很少有人在调试程序之外去直接改动某个类的功能。
四、结束语:
method swizzling就介绍这么多吧,希望能够帮助到大家。
Java反射(Method)(三) 日常开发中,我们可使用反射获取JavaBean的方法信息,而描述属性信息的就是Java Method,Method类同样位于java.lang.reflect包下。
Java反射之Method对象详解 使用Java反射,可以在运行时检查一个方法的信息以及在运行期调用这个方法,通过使用java.lang.reflect.Method类就可以实现上述功能。
由于最近开发新版本,就避免不了在开发和调试过程中引起崩溃,以及诱发一些之前的bug导致的崩溃。而且项目比较大也很不好排查,正好想起之前研究过的`Method Swizzling`,考虑是否能用这个苹果的“黑魔法”解决问题,当然用好这个黑魔法并不局限于解决这些问题.
相关文章
- IIS7.5上的REST服务的Put操作发生HTTP Error 405.0 - Method Not Allowed 解决方法
- C#设计模式——工厂方法模式(Factory Method Pattern)
- C#设计模式——工厂方法模式(Factory Method Pattern)
- C#设计模式——模板方法(Template Method)
- 【java设计模式】之 模板方法(Template Method)模式
- Atitit.研发管理--提升效率--软件开发方法DSM总结o99
- vue项目elementui 的 el-checkbox复选框改成单选框样式:实现全选、全否、单选、反选功能(最简单的方法)
- iOS:runtime 消息转发 Method_Swizzling 动态添加方法 动态方法解析
- Algorithm之MC:Monte Carlo method蒙特·卡罗方法的简介、实现、应用
- Py之openpyxl:openpyxl库的简介、安装、使用方法之详细攻略
- jQuery UI API 类别 - 方法重载(Method Overrides)
- (四)js数组方法一
- Android音量设置流程及调试方法
- java入门 常见问题0010:方法名不能望文生义,容易引起误解
- 按键消抖的两种方法--中断延迟工作与定时器