之前到别人的一篇博客上评论了下他的打点统计方法,后来很多人来问我,,所以还是决定写下这篇文章。第一次写博客,不喜勿喷。
关于统计打点,个人觉得最好的还是用面向切面编程思想(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函数获取所有的类,欢迎联系我,,解决这一问题。
网友评论
by 伍雪颖
int numClasses; Class *classes = NULL;
classes = NULL;
numClasses = objc_getClassList(NULL,0);
NSLog(@"Number of classes: %d", numClasses);
if (numClasses >0 )
{
classes = (__unsafe_unretained Class *)malloc(sizeof(Class) * numClasses);
numClasses = objc_getClassList(classes, numClasses);
for (int i = 0; i < numClasses; i++) {
NSLog(@"Class name: %s",class_getName(classes[i]));
}
free(classes); }