运行时runtime深度解析(二)—— Method Swizzling在页面统计上的应用

版本记录

版本号 时间
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

后记

未完,待续~~~~

风光无限好
最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
平台声明:文章内容(如有图片或视频亦包括在内)由作者上传并发布,文章内容仅代表作者本人观点,简书系信息发布平台,仅提供信息存储服务。

推荐阅读更多精彩内容