iOS 无感知埋点的原理

iOS开发中,一般使用hook的方式实现无感知埋点,hook过程一般在load方法中,通过方法的exchange来实现。

1. 关于 load 方法

类的+ (void)load 方法的加载发生在main函数之前,即pre-main阶段。因此,load方法中的逻辑要尽可能的简单,尽量不影响到APP的启动速度。

如果父类、子类、Category都实现了load方法,load的执行顺序是什么呢?
答:父类 > 子类 > Category分类。

如果有父类/子类有多个Category分类,那么这多个Category分类的load的执行顺序是什么呢?
答:编译资源的顺序决定了Category的执行顺序。可在工程配置的Build Phases选项中,设置Compile Sources中拖动分类的编译顺序。

如果有多个父类/子类,且有多个Category分类呢,那么这多个父类/子类、多个Category分类的load的执行顺序是什么呢?
答:先所有的父类、子类的load,然后再执行分类的load。

  1. +load方法是在加载类和分类时系统调用,一般不手动调用,如果想要在类或分类加载时做一些事情,可以重写类或者分类的+load方法方法;
  2. 每个类、分类的+load,在程序运行过程中只调用一次;
  1. 类要优先于分类调用+load方法;
  2. 子类调用+load方法时,要先要调用父类的+load方法;(父类优先与子类,与继承不同);
  3. 不同的类按照编译先后顺序调用+load方法(先编译,先调用);
  4. 分类的按照编译先后顺序调用+load方法(先编译,先调用)。

2. load 特殊执行顺序的原因

在 runtime 底层,会调用 prepare_load_methods 方法来准备好要被调用的 load 方法,具体方法实现:

void prepare_load_methods(const headerType *mhdr)
{
    size_t count, i;
    runtimeLock.assertWriting();
    classref_t *classlist = 
        _getObjc2NonlazyClassList(mhdr, &count);
    for (i = 0; i < count; i++) {
        schedule_class_load(remapClass(classlist[i]));
    }
    category_t **categorylist = _getObjc2NonlazyCategoryList(mhdr, &count);
    for (i = 0; i < count; i++) {
        category_t *cat = categorylist[i];
        Class cls = remapClass(cat->cls);
        if (!cls) continue;  // category for ignored weak-linked class
        realizeClass(cls);
        assert(cls->ISA()->isRealized());
        add_category_to_loadable_list(cat);
    }
}
//// 其中:
//classref_t *classlist = _getObjc2NonlazyClassList(mhdr, &count); //类列表
//category_t **categorylist = _getObjc2NonlazyCategoryList(mhdr, &count); //分类列表

static void schedule_class_load(Class cls)
{
if (!cls) return;
assert(cls->isRealized());  // _read_images should realize

if (cls->data()->flags & RW_LOADED) return;

// Ensure superclass-first ordering
schedule_class_load(cls->superclass);

add_class_to_loadable_list(cls);
cls->setInfo(RW_LOADED); 
}
//// 其中:
// schedule_class_load(cls->superclass); //在调度类的load方法前,要先跳用父类的load方法(递归),决定了父类优先于子类调用
// add_class_to_loadable_list(cls);  //添加到能够加载的类的列表中

void call_load_methods(void)
{
static bool loading = NO;
bool more_categories;

loadMethodLock.assertLocked();

// Re-entrant calls do nothing; the outermost call will finish the job.
if (loading) return;
loading = YES;

void *pool = objc_autoreleasePoolPush();

do {
// 1. Repeatedly call class +loads until there aren't any more
while (loadable_classes_used > 0) {
call_class_loads();
}

// 2. Call category +loads ONCE
more_categories = call_category_loads();

// 3. Run more +loads if there are classes OR more untried categories
} while (loadable_classes_used > 0  ||  more_categories);

objc_autoreleasePoolPop(pool);

loading = NO;
}

当prepare_load_methods函数执行完之后,所有满足+load方法调用条件的类和分类就被分别保持在全局变量中;

当prepare_load_methods执行完,准备好类和分类后,就该调用他们的+load方法啦,在call_load_methods中进行调用;注意图中红色圈内部分,两个关键函数:call_class_loads()、call_category_loads() ,就是这两个函数决定了类优先与分类调用+load方法;

说明:+load方法是系统根据方法地址直接调用,并不是objc_msgSend函数调用(isa,superClass);这就决定了如果子类没有实现+load方法,那么当它被加载时runtime是不会调用父类的+load方法的,除非父类也实现了+load方法;

load、initialize方法的区别

  • 调用方式
    load是根据函数地址直接调用
    initialize是通过objc_msgSend调用

  • 调用时刻
    load是runtime加载类、分类的时候调用(只会调用1次)
    initialize是类第一次接收到消息的时候调用,每一个类只会initialize一次(父类的initialize方法可能会被调用多次)

  • 调用顺序
    load:
    先调用类的load
    先编译的类,优先调用load
    调用子类的load之前,会先调用父类的load
    再调用分类的load
    先编译的分类,优先调用load
    initialize:
    先初始化父类
    再初始化子类(可能最终调用的是父类的initialize方法)

3. 方法的交换

交换系统方法也属于runtime的一部分,需要导入<objc/runtime.h>。
取出系统方法与你写的方法

#import <objc/runtime.h>

// 取出系统方法与自定义的方法
Method systemMethod = class_getInstanceMethod(self, @selector(systemMethod));
Method my_Method = class_getInstanceMethod(self, @selector(my_Method));

// 方法的交换
method_exchangeImplementations(systemMethod, my_Method);

一般交换的过程放在load中,如交换系统方法layoutSubviews与自定义的my_layoutSubviews,过程如下:

+ (void)load {
  Method systemMethod = class_getInstanceMethod(self, @selector(layoutSubviews));
  Method my_Method = class_getInstanceMethod(self, @selector(my_layoutSubviews));
  method_exchangeImplementations(systemMethod, my_Method);
}

- (void)layoutSubviews {
  [super layoutSubviews];
}
    
- (void)my_layoutSubviews {
        
  // 如果这么写,调用的就是my_layoutSubviews.就会循环引用.
  //[self layoutSubviews];
        
  // 正确写法
  [self my_layoutSubviews];
}

4. 埋点

我想你已经猜到该如何埋点了。通过上述一番操作,基本可以对工程里的类进行无感知拦截,并在自定义的交换方法中,获取及记录相关信息,然后择时上报。

load方法的处理
由于load是NSObject的方法,因此我们可以对UIControl、UITablview、UITapGesture、UIViewController等任何类去实现他们的分类,从而hook相关方法,去拦截事件、pv等统计的点。

如,UIControl的sendAction方法,UITablview的代理方法didSelected等;
如,对UITapGesturehook其中的初始化方法,并在自定义的初始化方法中,再次hook其传入的action对应的originalSEL,将其交换为自定义的目标action,并在其中统计埋点。

load方法的注意事项
一般需要确保只调用一次交换:

+(void)load
{
static dispatch_once_t onceToken;
dispatch_once(&onceToken, ^{
    // 交换操作
});
}

为不影响原代码调用逻辑,交换过的自定义方法中仍然要调用原方法:

- (void)my_layoutSubviews {
        
  // 如果这么写,调用的就是my_layoutSubviews.就会循环引用.
  //[self layoutSubviews];
        
  // 正确写法
  [self my_layoutSubviews];
}

埋点策略
首先是设计数据结构,一般是一个log后台统一的Json结构;其次是择时上报,根据log后台的上传格式、上传世纪策略准确处理文件;最后上报。

最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 214,837评论 6 496
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 91,551评论 3 389
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 160,417评论 0 350
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 57,448评论 1 288
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 66,524评论 6 386
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 50,554评论 1 293
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 39,569评论 3 414
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 38,316评论 0 270
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 44,766评论 1 307
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 37,077评论 2 330
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 39,240评论 1 343
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 34,912评论 5 338
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 40,560评论 3 322
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 31,176评论 0 21
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 32,425评论 1 268
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 47,114评论 2 366
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 44,114评论 2 352

推荐阅读更多精彩内容