iOS無(wú)侵入埋點(diǎn)方案

在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)原理圖:


4349969-6fa084ae3aac0b1f.png

具體的實(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

最后編輯于
?著作權(quán)歸作者所有,轉(zhuǎn)載或內(nèi)容合作請(qǐng)聯(lián)系作者
  • 序言:七十年代末津坑,一起剝皮案震驚了整個(gè)濱河市妙蔗,隨后出現(xiàn)的幾起案子,更是在濱河造成了極大的恐慌疆瑰,老刑警劉巖灭必,帶你破解...
    沈念sama閱讀 217,406評(píng)論 6 503
  • 序言:濱河連續(xù)發(fā)生了三起死亡事件,死亡現(xiàn)場(chǎng)離奇詭異乃摹,居然都是意外死亡禁漓,警方通過(guò)查閱死者的電腦和手機(jī),發(fā)現(xiàn)死者居然都...
    沈念sama閱讀 92,732評(píng)論 3 393
  • 文/潘曉璐 我一進(jìn)店門(mén)孵睬,熙熙樓的掌柜王于貴愁眉苦臉地迎上來(lái)播歼,“玉大人,你說(shuō)我怎么就攤上這事∶啬” “怎么了叭莫?”我有些...
    開(kāi)封第一講書(shū)人閱讀 163,711評(píng)論 0 353
  • 文/不壞的土叔 我叫張陵,是天一觀的道長(zhǎng)烁试。 經(jīng)常有香客問(wèn)我雇初,道長(zhǎng),這世上最難降的妖魔是什么减响? 我笑而不...
    開(kāi)封第一講書(shū)人閱讀 58,380評(píng)論 1 293
  • 正文 為了忘掉前任靖诗,我火速辦了婚禮,結(jié)果婚禮上支示,老公的妹妹穿的比我還像新娘刊橘。我一直安慰自己,他們只是感情好颂鸿,可當(dāng)我...
    茶點(diǎn)故事閱讀 67,432評(píng)論 6 392
  • 文/花漫 我一把揭開(kāi)白布促绵。 她就那樣靜靜地躺著,像睡著了一般嘴纺。 火紅的嫁衣襯著肌膚如雪败晴。 梳的紋絲不亂的頭發(fā)上,一...
    開(kāi)封第一講書(shū)人閱讀 51,301評(píng)論 1 301
  • 那天栽渴,我揣著相機(jī)與錄音位衩,去河邊找鬼。 笑死熔萧,一個(gè)胖子當(dāng)著我的面吹牛,可吹牛的內(nèi)容都是我干的僚祷。 我是一名探鬼主播佛致,決...
    沈念sama閱讀 40,145評(píng)論 3 418
  • 文/蒼蘭香墨 我猛地睜開(kāi)眼,長(zhǎng)吁一口氣:“原來(lái)是場(chǎng)噩夢(mèng)啊……” “哼辙谜!你這毒婦竟也來(lái)了俺榆?” 一聲冷哼從身側(cè)響起,我...
    開(kāi)封第一講書(shū)人閱讀 39,008評(píng)論 0 276
  • 序言:老撾萬(wàn)榮一對(duì)情侶失蹤装哆,失蹤者是張志新(化名)和其女友劉穎罐脊,沒(méi)想到半個(gè)月后,有當(dāng)?shù)厝嗽跇?shù)林里發(fā)現(xiàn)了一具尸體蜕琴,經(jīng)...
    沈念sama閱讀 45,443評(píng)論 1 314
  • 正文 獨(dú)居荒郊野嶺守林人離奇死亡萍桌,尸身上長(zhǎng)有42處帶血的膿包…… 初始之章·張勛 以下內(nèi)容為張勛視角 年9月15日...
    茶點(diǎn)故事閱讀 37,649評(píng)論 3 334
  • 正文 我和宋清朗相戀三年,在試婚紗的時(shí)候發(fā)現(xiàn)自己被綠了凌简。 大學(xué)時(shí)的朋友給我發(fā)了我未婚夫和他白月光在一起吃飯的照片上炎。...
    茶點(diǎn)故事閱讀 39,795評(píng)論 1 347
  • 序言:一個(gè)原本活蹦亂跳的男人離奇死亡,死狀恐怖雏搂,靈堂內(nèi)的尸體忽然破棺而出藕施,到底是詐尸還是另有隱情寇损,我是刑警寧澤,帶...
    沈念sama閱讀 35,501評(píng)論 5 345
  • 正文 年R本政府宣布裳食,位于F島的核電站矛市,受9級(jí)特大地震影響,放射性物質(zhì)發(fā)生泄漏诲祸。R本人自食惡果不足惜浊吏,卻給世界環(huán)境...
    茶點(diǎn)故事閱讀 41,119評(píng)論 3 328
  • 文/蒙蒙 一、第九天 我趴在偏房一處隱蔽的房頂上張望烦绳。 院中可真熱鬧卿捎,春花似錦、人聲如沸径密。這莊子的主人今日做“春日...
    開(kāi)封第一講書(shū)人閱讀 31,731評(píng)論 0 22
  • 文/蒼蘭香墨 我抬頭看了看天上的太陽(yáng)享扔。三九已至底桂,卻和暖如春,著一層夾襖步出監(jiān)牢的瞬間惧眠,已是汗流浹背籽懦。 一陣腳步聲響...
    開(kāi)封第一講書(shū)人閱讀 32,865評(píng)論 1 269
  • 我被黑心中介騙來(lái)泰國(guó)打工, 沒(méi)想到剛下飛機(jī)就差點(diǎn)兒被人妖公主榨干…… 1. 我叫王不留氛魁,地道東北人暮顺。 一個(gè)月前我還...
    沈念sama閱讀 47,899評(píng)論 2 370
  • 正文 我出身青樓,卻偏偏與公主長(zhǎng)得像秀存,于是被迫代替她去往敵國(guó)和親捶码。 傳聞我的和親對(duì)象是個(gè)殘疾皇子,可洞房花燭夜當(dāng)晚...
    茶點(diǎn)故事閱讀 44,724評(píng)論 2 354

推薦閱讀更多精彩內(nèi)容

  • GitHub項(xiàng)目地址 前言 最近業(yè)務(wù)需要加入一大批埋點(diǎn)統(tǒng)計(jì)事件或链,這個(gè)頁(yè)面添加一點(diǎn)代碼那個(gè)頁(yè)面添加一點(diǎn)代碼惫恼,各個(gè)頁(yè)面...
    青年別來(lái)無(wú)恙閱讀 1,981評(píng)論 0 23
  • 原文鏈接:無(wú)侵入的埋點(diǎn)方案如何實(shí)現(xiàn)? 前言: 原文中介紹了iOS開(kāi)發(fā)常見(jiàn)的埋點(diǎn)方式:代碼埋點(diǎn)澳盐、可視化埋點(diǎn)和無(wú)埋點(diǎn)祈纯。...
    YYYYYY25閱讀 1,644評(píng)論 3 16
  • Swift1> Swift和OC的區(qū)別1.1> Swift沒(méi)有地址/指針的概念1.2> 泛型1.3> 類(lèi)型嚴(yán)謹(jǐn) 對(duì)...
    cosWriter閱讀 11,100評(píng)論 1 32
  • 大家是希望文昌快點(diǎn)出場(chǎng),還是希望慢一點(diǎn)出場(chǎng)叼耙。
    知己伴一生閱讀 313評(píng)論 3 1
  • 有些日子不曾打電話給她了腕窥。 上次打電話回去的時(shí)候,因?yàn)橛行╇y過(guò)筛婉,想找她說(shuō)一說(shuō)油昂,結(jié)果卻是更難過(guò)。她不愛(ài)我么?她不能解...
    墨淡花已開(kāi)閱讀 147評(píng)論 0 0