美文网首页基础
无痕埋点实践

无痕埋点实践

作者: David_Cap | 来源:发表于2018-09-30 14:39 被阅读59次

参考了一些实现方案,自己选择了一种适合自己项目的,这里我将进行总结。

主体思路

主要思路就是要通过方法交换实现。

无痕埋点实践.png

拿Button举例,我们要确定一个可以路径来区分Button。
那么我这里使用的是:Target + Action + Tag
其实Target + Action 就可以确定一个唯一的Button。
Tag 是为了作为条件判断。例如:

- (void)buttonClick
{
    if(XXX)
    {
        //发送埋点1
    }
    else
    {
        //发送埋点2
    }
}

这样便可以使用Tag用于区分。

工具类

@implementation MethodSwizzingTool
+(void)swizzingForClass:(Class)cls originalSel:(SEL)originalSelector swizzingSel:(SEL)swizzingSelector
{
    Class class = cls;
    Method originalMethod = class_getInstanceMethod(class, originalSelector);
    Method  swizzingMethod = class_getInstanceMethod(class, swizzingSelector);
    
    BOOL addMethod = class_addMethod(class,
                                     originalSelector,
                                     method_getImplementation(swizzingMethod),
                                     method_getTypeEncoding(swizzingMethod));
    
    if (addMethod) {
        class_replaceMethod(class,
                            swizzingSelector,
                            method_getImplementation(originalMethod),
                            method_getTypeEncoding(originalMethod));
    }else{
        
        method_exchangeImplementations(originalMethod, swizzingMethod);
    }
}
@end

附加知识点

class_addMethod(Class,SEL,IMP,Type)
Class:添加方法到哪一个类
SEL:相当于方法的名称
IMP:IMP指针指向方法的具体实现
Type: 返回值类型

UIButton

UIButton继承于UIControl,每次的点击事件都会调用sendAction:to:forEvent:
核心代码如下:

@implementation UIControl (Analysis)

+ (void)load
{
    static dispatch_once_t onceToken;
    dispatch_once(&onceToken, ^{
        SEL originSelector = @selector(sendAction:to:forEvent:);
        SEL swizzingSelector = @selector(analysis_sendAction:to:forEvent:);
        
        [MethodSwizzingTool swizzingForClass:[self class] originalSel:originSelector swizzingSel:swizzingSelector];
    });
}

- (void)analysis_sendAction:(SEL)action to:(id)target forEvent:(UIEvent *)event
{
    [self analysis_sendAction:action to:target forEvent:event];
    
    //发送埋点
    ...
    ...
    ...
}
@end

UITableView

注释在代码中

@implementation UITableView (Analysis)

+ (void)load
{
    static dispatch_once_t onceToken;
    dispatch_once(&onceToken, ^{
        //交换setDelegate方法
        SEL originSelector = @selector(setDelegate:);
        SEL swizzingSelector = @selector(analysis_setDelegate:);
        [MethodSwizzingTool swizzingForClass:[self class] originalSel:originSelector swizzingSel:swizzingSelector];
    });
}


- (void)analysis_setDelegate:(id)delegate
{
    [self analysis_setDelegate:delegate];
    if (delegate) {
        SEL originSelector = @selector(tableView:didSelectRowAtIndexPath:);
        SEL swizzingSelector = @selector(analysis_tableView:didSelectRowAtIndexPath:);

        //重点
        //给Delegate添加的类 添加一个方法,方法的IMP指针是 self的analysis_tableView:didSelectRowAtIndexPath:
        //举个例子:Delegate是UIViewCobtroller,那么就是给UIViewController添加了一个 名字叫swizzingSelector的方法,方法的实现地址是self的analysis_tableView:didSelectRowAtIndexPath:
        BOOL addMethod = class_addMethod([delegate class],
                        swizzingSelector,
                        method_getImplementation(class_getInstanceMethod([self class], @selector(analysis_tableView:didSelectRowAtIndexPath:))),
                        nil);

        if (addMethod) {
            [MethodSwizzingTool swizzingForClass:[delegate class] originalSel:originSelector swizzingSel:swizzingSelector];
        }
    }

}

- (void)analysis_tableView:(UITableView *)tableView didSelectRowAtIndexPath:(NSIndexPath *)indexPath
{
    [self analysis_tableView:tableView didSelectRowAtIndexPath:indexPath];
    
    //发送埋点
}

@end

基本上其他的类似,这里不细讲。

维护json文件来发送埋点

如果在上述 //发送埋点的地方,添加一个统一的方法调用,这样就可以发送埋点数据。

- (void)sendEventWithTarget:(NSString *)target action:(NSString *)action index:(NSUInteger)index
{
    if (target.length == 0 || action.length == 0) {
        return;
    }
    
    if (![[self.analysisDictonary objectForKey:target] isKindOfClass:[NSDictionary class]])
    {
        return;
    }
    
    if (![[[self.analysisDictonary objectForKey:target] objectForKey:action] isKindOfClass:[NSArray class]]) {
        return;
    }
    
    NSArray *eventArray = [[self.analysisDictonary objectForKey:target] objectForKey:action];
    
    if (index < eventArray.count && [eventArray[index] isKindOfClass:[NSDictionary class]]) {
        NSDictionary *event = eventArray[index];
        if ([[event objectForKey:@"eventID"] isKindOfClass:[NSString class]]) {
            NSString *eventID = [event objectForKey:@"eventID"];
            NSLog(@"===EventID %@===",eventID);
            //发送埋点

        }
    }
}

这个json数据文件格式可以不相同,只要保证能找到数据就好。

进过如上的操作之后,那么其实之后的埋点添加就比较方便了,我们只要维护一份json文件就好,而且可以在点击的时候通过断点得知需要的信息,配置在json文件里面。

Demo

相关文章

  • 无痕埋点实践

    参考了一些实现方案,自己选择了一种适合自己项目的,这里我将进行总结。 主体思路 主要思路就是要通过方法交换实现。 ...

  • iOS无痕埋点方案分享探究

    iOS无痕埋点方案分享探究 iOS无痕埋点方案分享探究

  • web 埋点

    数据埋点是什么?设置数据埋点的意义?web 埋点实现原理了解一下 前端监控和前端埋点方案设计美团点评前端无痕埋点实践

  • 美团点评前端无痕埋点实践

    美团点评前端无痕埋点实践 -https://tech.meituan.com/mt-mobile-analytic...

  • 无痕埋点

    一、概念 通过技术手段无差别地记录用户在前端页面上的行为。可以正确的获取 PV、UV、IP、Action、Time...

  • ios 无痕埋点,两种方式

    FNKTrack 两种无痕埋点方式 FNKAopTrack 用Aspects进行AOP无痕埋点,具体见代码. FN...

  • iOS埋点之无痕埋点实践

    1、背景 稀里哗啦一大段 2、主要功能划分 从整个流程来说,我把他划分为下面几个主要的功能,事件拦截、viewPa...

  • ASM 无痕埋点

    ASM 是一个 Java 字节码操控框架。它能被用来动态生成类或者增强既有类的功能。ASM 可以直接产生二进制 c...

  • Asm初探

    最近项目中产品要求接入神策埋点,神策最大的宣传点应该就是所谓无痕全埋点。对于这种"无痕"或者"无感知",大部分An...

  • Android全链路无痕埋点实践

    背景 对无线开发同学而言,不管是支持业务数据采集,还是构建自动化测试体系,无痕埋点都是非常关键的技术路径之一。 目...

网友评论

    本文标题:无痕埋点实践

    本文链接:https://www.haomeiwen.com/subject/fuswoftx.html