對于一個移動App來說,統(tǒng)計用戶的使用習(xí)慣已經(jīng)是一個最基本的需求了。本文要講的不是教你如何去實現(xiàn)一個統(tǒng)計模塊,畢竟大部分的公司不會自己去開發(fā)一套統(tǒng)計系統(tǒng)。這里要講的是如何方便優(yōu)雅的集成第三方的統(tǒng)計系統(tǒng),如:友盟等。
一般的埋點都是每個要統(tǒng)計的地方都加一句埋點代碼,你剛畢業(yè)的時候這么寫沒有問題,但是如果你做了幾年的軟件開發(fā),還這樣寫,那就是你的錯了。這樣分散的代碼,不易于修改,不易于維護,還不易于測試。刪除添加什么的,都很麻煩。
所以,做為一個有點經(jīng)驗的開發(fā),我們應(yīng)該要想有沒有其它的方法來實現(xiàn)。一般的埋點就是在方法的開頭或者結(jié)尾插入一句統(tǒng)計的代碼。它不影響方法的運行?;谶@樣的要求。自然而然的就會想到,這是對方法進行AOP了。在iOS里面可以通過runtime來實現(xiàn)。也有一些很成熟的第三方庫,像Aspects,](https://github.com/steipete/Aspects),) 下面的例子就是通過Aspects將埋點代碼進行統(tǒng)一。
Aspects 給NSObject類添加了一個分類來做AOP:
+ (id<AspectToken>)aspect_hookSelector:(SEL)selector withOptions:(AspectOptions)options usingBlock:(id)block error:(NSError **)error;
假設(shè)我們有一個UserViewController,我們要統(tǒng)計它的打開次數(shù),可以這樣寫:
[UserViewController aspect_hookSelector:@selector(viewWillAppear:) withOptions:AspectPositionAfter usingBlock:^(id<AspectInfo> aspectInfo, NSString *code) {
// 這里寫友盟的統(tǒng)計代碼
} error:nil];
上面這句代碼是跟UserViewController的實例無關(guān)的,所以,我們可以把這句代碼抽離到一個新的類中。
#import "UmengHelper.h"
@implementation UmengHelper
+ (void)statistics {
[UserViewController aspect_hookSelector:@selector(viewWillAppear:) withOptions:AspectPositionAfter usingBlock:^(id<AspectInfo> aspectInfo) {
// 這里寫友盟的統(tǒng)計代碼
} error:nil];
[UserViewController aspect_hookSelector:@selector(addFriend:) withOptions:AspectPositionAfter usingBlock:^(id<AspectInfo> aspectInfo) {
// 這里寫友盟的統(tǒng)計代碼
} error:nil];
}
@end
上面這樣寫,把所有要埋點的地方都寫到同一個文件中了。但是有一個問題,這個新的類要引入所有用到的類。想一想這也是可以繼續(xù)優(yōu)化的。類名用字符串的形式保存,通過runtime得到對應(yīng)的類。再調(diào)hook方法就解決了。
再進一步想一下,既然類名跟方法名都是一個字符串,那么這些字符串是不是可以保存到后臺,通過接口下發(fā)。這樣就實現(xiàn)了動態(tài)埋點。
但是產(chǎn)品經(jīng)理是不會讓我們這么舒服的,他肯定又會有這樣的需求,『打開這個商品詳情的時候要統(tǒng)計這個商品的名稱』,Aspects也是支持參數(shù)傳遞的,通過那個usingBlock。
學(xué)了幾年iOS,現(xiàn)在工作中偶爾寫一點iOS相關(guān)的經(jīng)驗文章。Objective-C,Swift,RxSwift等。歡迎查看往期文章,喜歡的可以點個關(guān)注。