在iOS項(xiàng)目開(kāi)發(fā)中狸膏,我們要收集用戶(hù)的行為信息以便對(duì)項(xiàng)目進(jìn)行分析統(tǒng)計(jì)沟饥,就需要在代碼中進(jìn)行埋點(diǎn)統(tǒng)計(jì)。
一环戈、通常的埋點(diǎn)方式分為三種:
1.代碼埋點(diǎn)
在具體事件收集處手動(dòng)插入代碼闷板,優(yōu)點(diǎn)是能準(zhǔn)確的收集到需要統(tǒng)計(jì)的信息,缺點(diǎn)是插入代碼工作量比較大院塞,耦合度太高遮晚,后期維護(hù)和管理比較麻煩
2.可視化埋點(diǎn)
就是將埋點(diǎn)增加和修改的工作可視化了,提升了增加和維護(hù)埋點(diǎn)的體驗(yàn)拦止∠厍玻可視化埋點(diǎn)并非完全拋棄了代碼埋點(diǎn)篙议,而是在代碼埋點(diǎn)的上層封裝的一套邏輯來(lái)代替手工埋點(diǎn)
3.無(wú)埋點(diǎn)
無(wú)埋點(diǎn)继低,并不是不需要進(jìn)行埋點(diǎn)镜硕,而是需要“全埋點(diǎn)”七扰,而且埋點(diǎn)代碼不會(huì)出現(xiàn)在業(yè)務(wù)代碼中,容易管理和維護(hù)籽御。它的缺點(diǎn)是方案成本比較高绘沉,而且后期解析也比較復(fù)雜
可視化埋點(diǎn)和無(wú)埋點(diǎn)都是屬于無(wú)侵入的埋點(diǎn)方案赃绊,因?yàn)樗鼈兌疾恍枰诖a中寫(xiě)入埋點(diǎn)的代碼榴徐。所以采用無(wú)侵入式的埋點(diǎn)方案的優(yōu)點(diǎn)有:
1.埋點(diǎn)代碼與業(yè)務(wù)代碼剝離守问,降低耦合性
2.埋點(diǎn)方法集中統(tǒng)一管理匀归,減少漏埋點(diǎn)的幾率
二、無(wú)侵入埋點(diǎn)方法的實(shí)現(xiàn):
1.用運(yùn)行時(shí)方法替換方法進(jìn)行埋點(diǎn)
iOS常見(jiàn)的三種埋點(diǎn)就是:進(jìn)入頁(yè)面的次數(shù)耗帕、頁(yè)面停留時(shí)間穆端、點(diǎn)擊事件統(tǒng)計(jì),對(duì)于這幾種常見(jiàn)的埋點(diǎn)仿便,我們可以運(yùn)用運(yùn)行時(shí)方法替換來(lái)進(jìn)行插入埋點(diǎn)代碼体啰,以實(shí)現(xiàn)無(wú)侵入的埋點(diǎn)方法。
實(shí)現(xiàn)原理圖:
具體的實(shí)現(xiàn)方法是:先寫(xiě)一個(gè)運(yùn)行時(shí)方法替換的類(lèi) SMHook嗽仪,加上替換的方法
hookClass:fromSelector:toSelector,代碼如下:
#import "SMHook.h"
#import <objc/runtime.h>
@implementation SMHook
+ (void)hookClass:(Class)classObject fromSelector:(SEL)fromSelector toSelector:(SEL)toSelector {
Class class = classObject;
// 得到被替換類(lèi)的實(shí)例方法
Method fromMethod = class_getInstanceMethod(class, fromSelector);
// 得到替換類(lèi)的實(shí)例方法
Method toMethod = class_getInstanceMethod(class, toSelector);
// class_addMethod 返回成功表示被替換的方法沒(méi)實(shí)現(xiàn)荒勇,然后會(huì)通過(guò) class_addMethod 方法先實(shí)現(xiàn);返回失敗則表示被替換方法已存在闻坚,可以直接進(jìn)行 IMP 指針交換
if(class_addMethod(class, fromSelector, method_getImplementation(toMethod), method_getTypeEncoding(toMethod))) {
// 進(jìn)行方法的替換
class_replaceMethod(class, toSelector, method_getImplementation(fromMethod), method_getTypeEncoding(fromMethod));
} else {
// 交換 IMP 指針
method_exchangeImplementations(fromMethod, toMethod);
}
}
@end
這個(gè)方法利用運(yùn)行時(shí)method_exchangeImplementations進(jìn)行交換枕屉,當(dāng)原方法被調(diào)用時(shí),就會(huì)hook到指定的新方法去執(zhí)行鲤氢。
頁(yè)面進(jìn)入次數(shù)、頁(yè)面停留時(shí)間都需要對(duì)UIViewController 生命周期進(jìn)行埋點(diǎn)西潘,你可以創(chuàng)建一個(gè) UIViewController 的 Category卷玉,代碼如下:
@implementation UIViewController (logger)
+ (void)load {
static dispatch_once_t onceToken;
dispatch_once(&onceToken, ^{
// 通過(guò) @selector 獲得被替換和替換方法的 SEL,作為 SMHook:hookClass:fromeSelector:toSelector 的參數(shù)傳入
SEL fromSelectorAppear = @selector(viewWillAppear:);
SEL toSelectorAppear = @selector(hook_viewWillAppear:);
[SMHook hookClass:self fromSelector:fromSelectorAppear toSelector:toSelectorAppear];
SEL fromSelectorDisappear = @selector(viewWillDisappear:);
SEL toSelectorDisappear = @selector(hook_viewWillDisappear:);
[SMHook hookClass:self fromSelector:fromSelectorDisappear toSelector:toSelectorDisappear];
});
}
- (void)hook_viewWillAppear:(BOOL)animated {
// 先執(zhí)行插入代碼喷市,再執(zhí)行原 viewWillAppear 方法
[self insertToViewWillAppear];
[self hook_viewWillAppear:animated];
}
- (void)hook_viewWillDisappear:(BOOL)animated {
// 執(zhí)行插入代碼相种,再執(zhí)行原 viewWillDisappear 方法
[self insertToViewWillDisappear];
[self hook_viewWillDisappear:animated];
}
- (void)insertToViewWillAppear {
// 在 ViewWillAppear 時(shí)進(jìn)行日志的埋點(diǎn)
[[[[SMLogger create]
message:[NSString stringWithFormat:@"%@ Appear",NSStringFromClass([self class])]]
classify:ProjectClassifyOperation]
save];
}
- (void)insertToViewWillDisappear {
// 在 ViewWillDisappear 時(shí)進(jìn)行日志的埋點(diǎn)
[[[[SMLogger create]
message:[NSString stringWithFormat:@"%@ Disappear",NSStringFromClass([self class])]]
classify:ProjectClassifyOperation]
save];
}
@end
可以看到,Category 在 +load() 方法里使用了SMHook 進(jìn)行方法替換品姓,在替換的方法里執(zhí)行需要埋點(diǎn)的方法 [self insertToViewWillAppear]寝并。這樣的話,每個(gè)UIViewController生命周期到了ViewWillAppear都會(huì)執(zhí)行insertToViewWillAppear方法腹备。
在這里邊衬潦,我們是通過(guò)類(lèi)名NSStringFromClass([self class])來(lái)區(qū)分不同的控制器的。
對(duì)于點(diǎn)擊事件來(lái)說(shuō)植酥,我們也可以通過(guò)運(yùn)行時(shí)替換方法的方式來(lái)進(jìn)行無(wú)侵入埋點(diǎn)镀岛。 找到點(diǎn)擊事件的方法sendAction:to:forEvent:,然后再+(void)load方法中使用SMHook替換新的方法友驮。代碼如下:
+ (void)load {
static dispatch_once_t onceToken;
dispatch_once(&onceToken, ^{
// 通過(guò) @selector 獲得被替換和替換方法的 SEL漂羊,作為 SMHook:hookClass:fromeSelector:toSelector 的參數(shù)傳入
SEL fromSelector = @selector(sendAction:to:forEvent:);
SEL toSelector = @selector(hook_sendAction:to:forEvent:);
[SMHook hookClass:self fromSelector:fromSelector toSelector:toSelector];
});
}
- (void)hook_sendAction:(SEL)action to:(id)target forEvent:(UIEvent *)event {
[self insertToSendAction:action to:target forEvent:event];
[self hook_sendAction:action to:target forEvent:event];
}
- (void)insertToSendAction:(SEL)action to:(id)target forEvent:(UIEvent *)event {
// 日志記錄
if ([[[event allTouches] anyObject] phase] == UITouchPhaseEnded) {
NSString *actionString = NSStringFromSelector(action);
NSString *targetName = NSStringFromClass([target class]);
[[[SMLogger create] message:[NSString stringWithFormat:@"%@ %@",targetName,actionString]] save];
}
}
和UIViewController生命周期埋點(diǎn)不同的是,一個(gè)類(lèi)中可能有許多不同的UIButton子類(lèi)卸留,相同的UIButton子類(lèi)在不同的視圖中的埋點(diǎn)也要區(qū)分出來(lái)走越,所以我們通過(guò)NSStringFromClass([target class]) + NSStringFromSelector(action) 來(lái)區(qū)別,即類(lèi)名加方法名的格式作為唯一標(biāo)志耻瑟。
除了UIViewController旨指、UIButton控件外赏酥,Cocoa框架的其他控件都可以使用這種方法來(lái)進(jìn)行無(wú)侵入埋點(diǎn)。
2.事件唯一標(biāo)識(shí)
運(yùn)用運(yùn)行時(shí)替換方法的方式淤毛,我們能hook住所有的OC方法今缚,能夠幫我們解決了絕大部分的埋點(diǎn)問(wèn)題。
但是這種方案的精度還不夠高低淡,僅僅通過(guò)“ 方法名+視圖類(lèi)名”拼接的標(biāo)識(shí)還不能夠區(qū)分開(kāi)姓言。比如一個(gè)視圖下面有很多相同的按鈕,響應(yīng)的是同一個(gè)事件蔗蹋,如果僅僅是通過(guò)方法名+視圖類(lèi)型來(lái)區(qū)分何荚,顯然是不能區(qū)分到具體的事件,所以可以多添加一項(xiàng)目猪杭,比如“方法名+視圖類(lèi)名+按鈕標(biāo)題/索引”餐塘,這樣就能通過(guò)這個(gè)標(biāo)識(shí),精確的識(shí)別到某一個(gè)事件了皂吮。
3.上報(bào)機(jī)制
在收集到埋點(diǎn)信息之后戒傻,會(huì)有一個(gè)上報(bào)到服務(wù)器進(jìn)行統(tǒng)計(jì)分析的機(jī)制,比如實(shí)時(shí)發(fā)送蜂筹、啟動(dòng)時(shí)發(fā)送需纳、最小間隔時(shí)間發(fā)送等。服務(wù)器在接收到這些數(shù)據(jù)信息之后艺挪,按自己整理的計(jì)算統(tǒng)計(jì)規(guī)則得出最后的數(shù)據(jù)報(bào)表不翩,提供給相關(guān)人員對(duì)項(xiàng)目進(jìn)行分析使用。
總結(jié):
雖然使用運(yùn)行時(shí)方法的替換實(shí)現(xiàn)了無(wú)侵入埋點(diǎn)麻裳,但是該方案也存在著唯一標(biāo)志難以維護(hù)和準(zhǔn)確性難以保證的缺點(diǎn)口蝠。所以無(wú)侵入埋點(diǎn)還有比較長(zhǎng)的路要走。
參考資料:
https://time.geekbang.org/column/article/87925
http://www.reibang.com/p/7cd80e8bf29b