iOS开发专题程序员iOS dev

iOS 关于统计打点

2016-04-14  本文已影响2598人  yy倚楼听风雨

    之前到别人的一篇博客上评论了下他的打点统计方法,后来很多人来问我,,所以还是决定写下这篇文章。第一次写博客,不喜勿喷。

    关于统计打点,个人觉得最好的还是用面向切面编程思想(AOP),这样可以实现把我们的统计打点功能提出来与一批对象进行隔离,这样与一批对象之间降低了耦合性。这里需要用到的就是Method Swizzling,如果不知道Method Swizzling,先百度一下。下面开始一一说:

一、页面(UIViewController)统计 

UIViewController统计比较简单,一般也就需要在viewWillAppear和viewWillDisappear里进行统计 。所以我们只需要在category中交换viewWillAppear和viewWillDisappear和两个方法即可

#import "UIViewController+Tracking.h"

#import <objc/runtime.h>

@implementation UIViewController (Tracking)

+ (void)load {

// 交换方法viewWillAppear:

method_exchangeImplementations(class_getInstanceMethod(self, @selector(viewWillAppear:)),class_getInstanceMethod(self, @selector(tracking_viewWillAppear:)));

//交换方法viewWillDisappear:

method_exchangeImplementations(class_getInstanceMethod(self, @selector(viewWillDisappear:)), class_getInstanceMethod(self, @selector(tracking_viewWillDisappear:)));

}

- (void)tracking_viewWillAppear:(BOOL)animated {

[self tracking_viewWillAppear:animated];

//此处添加你想统计的打点事件

NSLog(@"当前viewController :%@",NSStringFromClass([self class]));

}

- (void)tracking_viewWillDisappear:(BOOL)animated {

[self tracking_viewWillDisappear:animated];

//此处添加你想统计的打点事件

NSLog(@"当前viewController :%@",NSStringFromClass([self class]));

}

经过以上交换 ,所有的viewController就会走到自定义的tracking_viewWillAppear和tracking_viewWillDisappear方法,就可以在注释的地方做页面统计事件了。这里统计建议先建立个viewController的类名或者title作为索引值,自定义标识为键值的字典或plist文件,比如@{@"homeViewController" : @"首页"},然后就可以直接在注释处通过NSStringFromClass([self class])获取到viewController的类名然后索引到自定义标识。

不过有时候,可能会出现一个viewController可能会复用,比如用type(或title)区分的情况,,这时候,就需要你在注释处,对self进行判断,如果是这些类,强转后得到该类,在类名后拼接上type(或title)作为字典索引值,比如@{@"homeViewController_type1" : @"首页", @"homeViewController_type2" : @"第二页",}

二、按钮(UIControl)及UIBarButtonItem点击事件

要想统计UIControl点击事件,,首先要知道从哪地方进行方法交换,在UIControl里找到sendAction:to:forEvent:方法,,这是每次点击都会走的方法

#import "UIControl+Tracking.h"

#import <objc/runtime.h>

@implementation UIControl (Tracking)

+ (void)load

{

method_exchangeImplementations(class_getInstanceMethod(self, @selector(sendAction:to:forEvent:)), class_getInstanceMethod(self, @selector(tracking_sendAction:to:forEvent:)));

}

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

{

[self tracking_sendAction:action to:target forEvent:event];

//此处添加你想统计的打点事件

}

就这么两句代码 ,然后所有的UIControl点击事件,就会走这注释处,我们就可以进行点击统计了,在注释这里,可以获取到的信息有action(点击响应的方法),control的target,self(就是control本身了,可以获取title,tag等信息)还有event。简单吧。不过这里要建的索引表就有点复杂了,最常见的,在同一个target(比如viewController)下,多个control指向同一方法,这时我们就需要用self的tag属性来做区分了。所以我建议拼接索引值时将target,action和tag三个值给拼起来。如@{@"homeViewController_searchAction_tag1" : @"首页的第一个搜索按钮" , @"homeViewController_searchAction_tag2" : @"首页的第二个搜索按钮"}

UIBarButtonItem 的点击事件也会走到- (void)tracking_sendAction:(SEL)action to:(id)target forEvent:(UIEvent *)event方法这,只是target变成了UIBarButtonItem,所以我们需要到这方法里,对target进行判断

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

{

[self tracking_sendAction:action to:target forEvent:event];

if ([target isKindOfClass:[UIBarButtonItem class]]) {

UIBarButtonItem *item = (UIBarButtonItem *)target;

// 在此拼接NSStringFromClass([item.target class])和NSStringFromSelector(item.action)]]

}else {

//此处添加你想统计的打点事件

// 在此拼接将NSStringFromClass([target class]) ,NSStringFromSelector(action)和tag三个值给拼起来

}

}

三、UITableView的cell点击事件

UITableView的点击事件就稍微麻烦点了,因为需要切面的点击方法tableView:didSelectRowAtIndexPath:是在代理那,所以我们需要先替换设置代理方法,获取到代理人,然后再进行切面,上代码

#import "UITableView+Tracking.h"

#import <objc/runtime.h>

#import <objc/message.h>

@implementation UITableView (Tracking)

+ (void)load{   

 //交换实现setDelegate,获取到代理人    method_exchangeImplementations(class_getInstanceMethod(self, @selector(setDelegate:)), class_getInstanceMethod(self, @selector(tracking_setDelegate:)));

}

- (void)tracking_setDelegate:(id)delegate

{

[self tracking_setDelegate:delegate];

Class class = [delegate class];

// 在代理人这先添加用于实现统计的方法,然后和交换原先的点击方法

if (class_addMethod(class, NSSelectorFromString(@"tracking_didSelectRowAtIndexPath"), (IMP)tracking_didSelectRowAtIndexPath, "v@:@@")) {

Method dis_originalMethod = class_getInstanceMethod(class, NSSelectorFromString(@"tracking_didSelectRowAtIndexPath"));

Method dis_swizzledMethod = class_getInstanceMethod(class, @selector(tableView:didSelectRowAtIndexPath:));

//交换实现

method_exchangeImplementations(dis_originalMethod, dis_swizzledMethod);

}

}

void tracking_didSelectRowAtIndexPath(id self, SEL _cmd, id tableView, id indexpath)

{

SEL selector = NSSelectorFromString(@"tracking_didSelectRowAtIndexPath");

((void(*)(id, SEL,id, id))objc_msgSend)(self, selector, tableView, indexpath);

//此处添加你想统计的打点事件

}

@end

这里会因为要将代理人的方法进行切面,所以这需要给代理人通过class_addMethod这方式动态的添加方法。至于为什么要判断是因为,只能第一次添加时才进行交换,不判断的话,多设置几次delegate,就会多交换几次,偶数次就会还原了,不会进入设置好的统计打点方法。至于为什么要写成((void(*)(id, SEL,id, id))objc_msgSend),是为什么兼容ios的多版本问题。最后的统计和上面的也差不多了,都是建立字典,只不过已知信息换成了(id self, SEL _cmd, id tableView, id indexpath),里面self也就是delegate,

四,UICollectionView的点击统计

UICollectionView的点击统计和UITableView的点击统计原理差不多,都是先交换setDelegate:这里就不展示了,想要看直接去我的github上下载就可以了,下载地址https://github.com/363128432/ActionTracking/tree/master/Tracking

最后,很多人还问,能不能自动生成索引字典的key,,这我只想到了进入每个页面时自动写成plist文件的key,最后导出成,,但是需要人一个页面一个页面创建,如果有谁知道怎么一开始在main函数获取所有的类,欢迎联系我,,解决这一问题。

上一篇 下一篇

猜你喜欢

热点阅读