成人免费xxxxx在线视频软件_久久精品久久久_亚洲国产精品久久久_天天色天天色_亚洲人成一区_欧美一级欧美三级在线观看

iOS 統計打點那些事

移動開發
單來說,就是可以動態的在函數調用的前后插一段代碼。iOS 可以使用 Pete Steinberger 開發的 Aspects 這個庫,大致原理是在 runtime 層,通過 swizzle method 來實現的。 來看一個小 Demo

[[148848]]

統計打點是 App 開發里很重要的一個環節,App 的運行狀態、改版后的效果、用戶的各種行為等都需要打點,市面上也有不少可供選擇的第三方庫。 假設產品有這么個需求:當用戶在詳情頁點擊購買按鈕時,記錄一下事件。我們實現起來大概會是這樣

  1. // DetailViewController.m 
  2.  
  3. - (void)onBuyButtonTapped:(UIButton *)button 
  4. // do some stuff, maybe send a request to server 
  5. [XXXAnalytics event:kSomeEventYouDefined]; 

這個需求就這樣輕松搞定了,但細細想想還是有不少問題的:

頁面上會有其他的 Button,可能每個 Button 都要放上這么一段代碼。

這些統計其實跟具體的業務無關,沒必要跟業務代碼混雜在一起,不優雅。

當改版或者重構時,有可能忘了把相應的打點代碼遷移過去。

所以需要一種更好的方式來做這件事,這就是使用 AOP(Aspect-Oriented-Programming),翻譯過來就是「面向切面編程」

通過預編譯方式和運行期動態代理實現在不修改源代碼的情況下給程序動態統一添加功能的一種技術。

簡單來說,就是可以動態的在函數調用的前后插一段代碼。iOS 可以使用 Pete Steinberger 開發的 Aspects 這個庫,大致原理是在 runtime 層,通過 swizzle method 來實現的。

來看一個小 Demo

  1. [UIViewController aspect_hookSelector:@selector(viewWillAppear:) withOptions:AspectPositionAfter usingBlock:^(idaspectInfo, BOOL animated) { 
  2. NSLog(@"View Controller %@ will appear animated: %tu", aspectInfo.instance, animated); 
  3. } error:NULL]; 

這樣在 UIViewController 的 viewWillAppear: 被調用后,還會再調一下我們定義的 Block,這段日志就會被輸出。而打點正好符合這種場景:正事干完之后,額外干一些跟業務無關的事情。

上面的例子,我們通過 AOP 來做的話,大概就是這樣

  1. // DetailViewController.m 
  2. - (void)onBuyButtonTapped:(UIButton *)button 
  3. // do some stuff, maybe send a request to server 
  4. // no need to call [XXXAnalytics event:] 
  5.  
  6. // AppDelegate.m 
  7. - (void)setupAnalytics 
  8. [DetailViewController aspect_hookSelector:@selector(onBuyButtonTapped:) withOptions:AspectPositionAfter usingBlock:^(idaspectInfo, BOOL animated) { 
  9. [XXXAnalytics event:kSomeEventYouDefined]; 
  10. } error:NULL]; 

這樣統計代碼就從業務代碼中剝離出來了。但是又產生了一個新問題,多個 Button Event,豈不是要寫很多行這樣的代碼,「重復」這樣的事情,作為一個程序員怎么能忍,簡單,造一個方法

  1. - (void)trackEventWithClass:(Class)klass selector:(SEL)selector event:(NSString *)event 
  2. [klass aspect_hookSelector:@selector(selector) withOptions:AspectPositionAfter usingBlock:^(idaspectInfo, BOOL animated) { 
  3. [XXXAnalytics event:event]; 
  4. } error:NULL]; 

使用起來就像這樣

  1. - (void)setupAnalytics 
  2. [self trackEventWithClass:DetailViewController selector:@seletor(onBuyButtonTapped:) event:kSomeEventYouDefined]; 
  3. [self trackEventWithClass:ListViewController selector:@seletor(followButtonTapped:) event:kAnotherEventYouDefined]; 
  4. // ... 

看起來又干凈了些。這時,產品經理又提了個需求:當這個按鈕點擊時,如果已經登錄了,發送 EventA,如果沒有登錄則發送 EventB,也就是說,不再只是 [XXXAnalytics event:] 這么簡單了,還需要加上額外的邏輯,這也難不倒我們,加上一個 block 即可。

  1. - (void)trackEventWithClass:(Class)klass 
  2. selector:(SEL)selector 
  3. eventHandler:(void (^)(idaspectInfo))eventHandler 
  4. [klass aspect_hookSelector:@selector(selector) withOptions:AspectPositionAfter usingBlock:^(idaspectInfo, BOOL animated) { 
  5. if (eventHandler) { 
  6. eventHandler(aspectInfo); 
  7. } error:NULL]; 
  8.  
  9. // 使用 
  10. [self trackEventWithClass:DetailViewController selector:@seletor(onBuyButtonTapped:) eventHandler:^(idaspectInfo){ 
  11. user.loggedIn ? [XXXAnalytics event:EventA] : [XXXAnalytics event:EventB]; 
  12. }]; 

好了,現在只要不是太復雜的打點邏輯(那些需要方法上下文變量的)我們都能應付了,接下來就該等產品來驗收了。產品搬了個凳子坐在身邊,然后點一下 Button,看一下 Console,被幾輪蹂躪后,產品也慢慢地接受了這種驗收方式。后來某一天,忽然發現某一項或某幾項數據有異常,然后找到開發,瞄了一眼:哦,這個方法被重構了。或者新加的方法忘了加統計了。只能等到下個版本再加上了,如果只是一般的統計數據倒還好,跟錢相關的就麻煩了。

那么有沒有一種直觀的驗證方式呢?當然,程序員是***的呀。一個理想的狀況是,產品打開 App 后,開啟某個開關就能看到所有會發送 Event 的按鈕,就像這樣

 

其中數字代表了 EventID。如何實現呢?還記得注冊事件時,我們有傳入 class 和 selector 么,一般我們都會有一個 BaseViewController,那么就可以在 BaseViewController 的 viewDidAppear: 里做點文章了。

  1. // BaseViewController.m 
  2. - (void)viewDidAppear:(BOOL)animated 
  3. [super viewDidAppear:animated]; 
  4. // 獲取已經注冊過的 classes 
  5. NSDictionary *registeredClasses = [OurAnalytics sharedInstance].registeredClasses; 
  6.  
  7. [registeredClasses enumerateKeysAndObjectsUsingBlock:^(NSString *className, NSArray *selectors, BOOL *stop) { 
  8. if ([self isKindOfClass:NSClassFromString(className)]) { 
  9. // 如何根據 selector 找到它的宿主? 
  10. }]; 

所以現在問題就剩下,如何根據 selector 找到對應的 Button,這里要注意,有些 Button 可能要等網絡請求完成才會出現,比如 TableViewCell 里的 Button。

沒有想到太方便的方法,簡單粗暴點就是設置個 Timer 每隔一段時間掃一下 subviews,如果是 button 或 包含 tapGesture 的,就拿它們的 action 對比一下,如果 match 就可以高亮那個 button / view 了。

EventID 也一樣,之前在注冊時也會傳一個 EventID 過來,這里直接顯示出來即可。對于那些傳 eventHandler 的就不行了。

所以理論上是可行的,性能上會稍微有點損耗,尤其是當 subViews 的結構比較復雜時,不過只是內部用來做驗證,所以這也不是什么問題。

看起來效果已經不錯了,有沒有可能讓這套體系再靈活一些?比如可以從后端制定打點規則?客戶端只是讀取一個配置文件,就像這樣

  1. - (void)setupAnalytics 
  2. // analyticsRules 是從配置文件中讀取出來的 
  3. [analyticsRules enumerateObjectsUsingBlock:^(NSDictionary *rules, NSUInteger idx, BOOL *stop) { 
  4. Class klass = NSClassFromString(rules[@"class"]); 
  5. SEL selector = NSSelectorFromString(rules[@"selector"]); 
  6. NSString *eventID = rules[@"eventID"]; 
  7. [self trackEventWithClass:klass seletor:seletor event: eventID]; 
  8. }]; 

那如果在后臺的時候填錯了 Class 或 Selector 怎么辦?還好有 objc_getClassList 和 class_copyMethodList 這兩個運行時方法,有了它們就可以在 App 啟動時掃一遍已注冊的類(過濾掉 UI / NS 開頭的),然后將它們的 seletor 也一并保存下來發送給服務端,當然這種操作只需在適當的時機做一下就可以了,比如集成打包時。

現在,這套體系就比較完整了。當然這只是我的一些構想,并沒有在實踐中嘗試過,所以肯定會踩到各種各樣的坑,不過至少看起來是個可行的方案。

責任編輯:chenqingxiang 來源: Limboy
相關推薦

2017-01-10 13:33:51

iOS編程throttle

2021-07-27 10:52:27

iOS WKWebView容器

2011-05-19 16:47:50

軟件測試

2012-05-01 08:06:49

手機

2017-05-15 21:50:54

Linux引號

2024-02-04 17:03:30

2015-05-28 14:02:09

JavaJava日志性

2011-08-22 16:42:43

SqliteiPad

2011-12-02 10:32:23

Java

2014-06-06 16:08:17

初志科技

2021-10-19 21:39:51

Unsafe構造器內存

2020-09-23 09:07:16

特權賬號管理PAM網絡安全

2011-09-19 15:40:35

2020-07-29 08:14:59

云計算云遷移IT

2009-07-29 10:36:04

北電收購

2012-01-02 19:30:22

iPad

2011-07-04 15:30:24

Qt 布局 GridLayout

2010-07-26 11:02:19

Perl模式匹配

2011-06-30 14:34:17

QT Tablewidge QTableWidg

2012-05-31 09:53:38

IT風云15年
點贊
收藏

51CTO技術棧公眾號

主站蜘蛛池模板: 国精产品一区一区三区免费完 | 97视频成人| 91色啪 | 国产欧美精品区一区二区三区 | 精品国产乱码久久久久久a丨 | 精品国产一二三区 | 欧美日韩国产一区二区三区 | 亚洲一区有码 | 久久6视频 | 久久久久久国产精品mv | 成人在线观看网址 | 91视频免费在观看 | 超碰91在线 | 天天欧美 | 国产精品中文字幕在线播放 | 欧美日韩精品专区 | 青青操av | 国产在线精品一区 | www.日韩| 在线久草| 日韩免费视频 | 国产你懂的在线观看 | 亚洲精品毛片av | 91精品国产综合久久久久久首页 | 久久久久九九九女人毛片 | 中文字幕高清av | 国产精品波多野结衣 | 成人免费影院 | 91在线视频免费观看 | 成人精品一区二区三区 | 一级欧美 | 综合色在线 | 国产精品地址 | 中文在线日韩 | 中文字幕二区 | 99久久久99久久国产片鸭王 | 欧美黄色精品 | 91在线最新 | 最新伦理片| 亚洲男人网 | 国产日韩欧美 |