版本记录
版本号 | 时间 |
---|---|
V1.0 | 2017.07.27 |
前言
OC是运行时的语言,底层就是运行时,可以说runtime是OC的底层,很多事情也都可以用运行时解决,下面就讲述一下运行时runtime的知识以及它的妙用。感兴趣的可以看上面几篇。
1. 运行时runtime深度解析(一)—— API
Method Swizzling
Method Swizzing
是发生在运行时的,主要用于在运行时将两个Method
进行交换,我们可以将Method Swizzling
代码写到任何地方,但是只有在这段Method Swilzzling
代码执行完毕之后互换才起作用。而且Method Swizzling
也是iOS中AOP
(面相切面编程)的一种实现方式,我们可以利用苹果这一特性来实现AOP
编程。
在OC语言的runtime特性中,调用一个对象的方法就是给这个对象发送消息。是通过查找接收消息对象的方法列表,从方法列表中查找对应的SEL,这个SEL对应着一个IMP(一个IMP可以对应多个SEL),通过这个IMP找到对应的方法调用。在每个类中都有一个Dispatch Table
,这个Dispatch Table本质是将类中的SEL和IMP(可以理解为函数指针)进行对应。而我们的Method Swizzling就是对这个table进行了操作,让SEL对应另一个IMP。
下面看其原理图。
1. 方法互换在页面统计上的应用需求
很多公司都有页面统计这个需求,这里我们也做一下统计,主要有两种思路:
- 在每一个控制器中
viewDidLoad
方法中统计用户进入控制器的次数,并上报至服务器。但是这有个缺点就是每一个控制器都要加很是繁琐。 - 还有一种办法就是写一个
UIViewController
的分类Category
,然后在Category
中的+(void)load
方法中添加Method Swizzling
方法,我们用来替换的方法也写在这个Category中。由于load类方法是程序运行时这个类被加载到内存中就调用的一个方法,执行比较早,并且不需要我们手动调用。而且这个方法具有唯一性,也就是只会被调用一次,不用担心资源抢夺的问题。
这里我们采用的是第二种方法,正好也验证下Method Swizzling
方法的使用。
2. 方法互换在页面统计上的应用实现
下面我们就直接看代码吧。
1. JJRuntimeVC.h
#import <UIKit/UIKit.h>
@interface JJRuntimeVC : UIViewController
@end
2. JJRuntimeVC.m
#import "JJRuntimeVC.h"
#import "UIViewController+JJSwizzlingCategory.h"
@interface JJRuntimeVC ()
@end
@implementation JJRuntimeVC
- (void)viewDidLoad
{
[super viewDidLoad];
self.view.backgroundColor = [UIColor greenColor];
}
@end
3. UIViewController+JJSwizzlingCategory.h
#import <UIKit/UIKit.h>
@interface UIViewController (JJSwizzlingCategory)
@end
4. UIViewController+JJSwizzlingCategory.m
#import "UIViewController+JJSwizzlingCategory.h"
#import <objc/runtime.h>
@implementation UIViewController (JJSwizzlingCategory)
#pragma mark - Override Base Function
+ (void)load
{
[super load];
//通过class_getInstanceMethod()函数从当前对象中的method list获取method结构体,如果是类方法就使用class_getClassMethod()函数获取。
Method replacedMathod = class_getInstanceMethod([self class], @selector(viewDidLoad));
Method toReplaceMethod = class_getInstanceMethod([self class], @selector(swizzlingMethodViewDidLoad));
// 我们在这里使用class_addMethod()函数对Method Swizzling做了一层验证,如果self没有实现被交换的方法,会导致失败。
// 而且self没有交换的方法实现,但是父类有这个方法,这样就会调用父类的方法,结果就不是我们想要的结果了。
// 所以我们在这里通过class_addMethod()的验证,如果self实现了这个方法,class_addMethod()函数将会返回NO,我们就可以对其进行交换了。
if (!class_addMethod([self class], @selector(viewDidLoad), method_getImplementation(toReplaceMethod), method_getTypeEncoding(toReplaceMethod))) {
method_exchangeImplementations(replacedMathod, toReplaceMethod);
}
}
#pragma mark - Action && Notification
- (void)swizzlingMethodViewDidLoad
{
NSString *str = [NSString stringWithFormat:@"%@",self.class];
// 我们在这里加一个判断,将系统的UIViewController的对象剔除掉
if (![str containsString:@"UI"]) {
NSLog(@"统计打点:%@",self.class);
}
[self swizzlingMethodViewDidLoad];
}
@end
运行代码会发现,先走+ (void)load
实现方法的互换,再走控制器JJRuntimeVC
中的viewDidLoad
方法,但是由于在+ (void)load
中对方法的实现做了互换,所以走的是方法- (void)swizzlingMethodViewDidLoad
,在这个方法内部接着调用[self swizzlingMethodViewDidLoad];
,同样由于方法实现的互换,其实调用的是方法- (void)viewDidLoad
,所以最后又走正常的走了这个方法,同时实现了对加载控制器的统计,下面看结果输出。
2017-07-27 19:07:27.892845+0800 JJOC[5763:1911605] 统计打点:JJRuntimeVC
这样就利用运行时实现了页面统计。
参考文献
1. Method Swizzling
2. Objective-C Runtime 运行时之四:Method Swizzling
3. iOS黑魔法-Method Swizzling
后记
未完,待续~~~~