Hugo 应用内方法调用监控实践

做项目的时候有时候需要打印方法的传参和返回值,甚至方法的执行时间,有没有一种简单方便通用的方式去做这个呢,Hugo就可以。
使用方法很简单,Hugo是基于注解被调用的,引入相关依赖后,在方法上加上 @DebugLog 即可, 也可以在类前加上@DebugLog, 对该类的所有方法都可以监控到。
Hugo这个项目使用起来其实非常简单, 但我对gradle编译不是太熟悉, 中间花了1天多的时间来解决编译上的问题。

方法总结如下:
新建一个工程myapplication
1. 只需要在build.gradle上加上这几句话就可以了, android studio会远程下载需要的依赖包.
buildscript {
    repositories {
        mavenCentral()
    }

    dependencies {
        classpath 'com.jakewharton.hugo:hugo-plugin:1.2.1'
    }
}

apply plugin: 'com.android.application'
apply plugin: 'com.jakewharton.hugo'
2. 在class前或是方法前加上注解@DebugLog
import hugo.weaving.DebugLog;

@DebugLog
public class MainActivity extends AppCompatActivity {
    @Override
    protected void onCreate(Bundle savedInstanceState) {
        super.onCreate(savedInstanceState);
        setContentView(R.layout.activity_main);
        getName("ahking","wang");
    }

    @Override
    protected void onPause() {
        super.onPause();
    }

    public String getName(String first, String last) {
        SystemClock.sleep(15); // Don't ever really do this!
        return first + " " + last;
    }
}
3. 在logcat中可以看到输出的log信息
08-05 10:41:23.677  10151-10151/com.example.myapplication V/MainActivity﹕ ⇢ <init>()
08-05 10:41:23.687  10151-10151/com.example.myapplication V/MainActivity﹕ ⇠ <init> [0ms]
08-05 10:41:23.687  10151-10151/com.example.myapplication V/MainActivity﹕ ⇢ onCreate(savedInstanceState=null)
08-05 10:41:23.768  10151-10151/com.example.myapplication V/MainActivity﹕ ⇢ getName(first="ahking", last="wang")
08-05 10:41:23.788  10151-10151/com.example.myapplication V/MainActivity﹕ ⇠ getName [15ms] = "ahking wang"
08-05 10:41:23.788  10151-10151/com.example.myapplication V/MainActivity﹕ ⇠ onCreate [102ms]
08-05 10:41:23.898  10151-10151/com.example.myapplication V/MainActivity﹕ ⇢ onCreateOptionsMenu(menu=android.support.v7.view.menu.MenuBuilder@41658010)
08-05 10:41:23.908  10151-10151/com.example.myapplication V/MainActivity﹕ ⇠ onCreateOptionsMenu [1ms] = true
08-05 10:41:32.286  10151-10151/com.example.myapplication V/MainActivity﹕ ⇢ onPause()
08-05 10:41:32.286  10151-10151/com.example.myapplication V/MainActivity﹕ ⇠ onPause [0ms]

以getName()的调用为例, 不仅输出了调用参数, 而且输出返回值, 以及这个方法的执行时间, 这对于调试性能问题是非常有帮助的.
完整的项目代码在: /home/wangxin/src/github/hugo/myapplication

应用在chromium项目上

把hugo应用在chromium项目上, 只在build.gradle上添加上面的那几句话, 编译是无法通过的。
通过在myapplication项目上查看@DebugLog的定义, 发现它的实现在这里:

/home/wangxin/.gradle/caches/modules-2/files-2.1/com.jakewharton.hugo/hugo-annotations/1.2.1/52d129a681468a4df976ff411fd163265dc6f99c/hugo-annotations-1.2.1-sources.jar

于是在chrome的build.gradle上连蒙带猜的添加:

dependencies {
    compile fileTree(dir: 'libs', exclude: 'android-support-multidex.jar', include: '*.jar')
    compile 'com.android.support:multidex:1.0.0'
    compile project(':mediaplayer')
    compile project(':web_contents_delegate_android')
    compile project(':browser_I')
    compile project(':chromium_gen')
    compile files('libs/hugo-annotations-1.2.1-sources.jar') //加上这行
    compile files('libs/decrawso.jar')
    compile files('libs/qihoospeechrecognition.jar')
    compile files('libs/QHStatAgent.jar')
    compile files('libs/adsdk_0.1.16.1125.jar')
    compile files('libs/andfix.jar')
    compile files('libs/opensdk-release.jar')
}
//需要把jar文件拷贝到 “/home/wangxin/src/src_chrome45ce_rel_v6/m_browser_chromium/chrome/android/java/libs”目录下

在build.gradle的末尾的{}外, 再加上下面这几行代码:

buildscript {
    repositories {
        mavenCentral()
    }

    dependencies {
        classpath 'com.jakewharton.hugo:hugo-plugin:1.2.1'
    }
}
apply plugin: 'com.jakewharton.hugo'
在ChromeTabbedActivity类前加上注解
import hugo.weaving.DebugLog;
@DebugLog
public class ChromeTabbedActivity extends ChromeActivity implements ActionBarDelegate,
        OverviewModeObserver, INetworkChangeListener, IOrientationListener {

Log输出如下:

08-05 14:36:32.502  21461-21461/com.qihoo.browser V/ChromeTabbedActivity﹕ ⇠ postInflationStartup [131ms]
08-05 14:36:32.512  21461-21461/com.qihoo.browser V/ChromeTabbedActivity﹕ ⇢ onStart()
08-05 14:36:32.512  21461-21461/com.qihoo.browser V/ChromeTabbedActivity﹕ ⇠ onStart [0ms]
08-05 14:36:32.512  21461-21461/com.qihoo.browser V/ChromeTabbedActivity﹕ ⇢ onResume()
08-05 14:36:32.522  21461-21461/com.qihoo.browser V/ChromeTabbedActivity﹕ ⇠ onResume [3ms]
08-05 14:36:32.963  21461-21461/com.qihoo.browser V/ChromeTabbedActivity﹕ ⇢ onCreateWithNative()
08-05 14:36:32.963  21461-21461/com.qihoo.browser V/ChromeTabbedActivity﹕ ⇢ AsyncLoadData()
08-05 14:36:32.963  21461-21461/com.qihoo.browser V/ChromeTabbedActivity﹕ ⇢ cacheFrequentFromDb()
08-05 14:36:32.983  21461-21461/com.qihoo.browser V/ChromeTabbedActivity﹕ ⇠ cacheFrequentFromDb [13ms]
08-05 14:36:32.983  21461-21461/com.qihoo.browser V/ChromeTabbedActivity﹕ ⇠ AsyncLoadData [13ms]
08-05 14:36:32.983  21461-21461/com.qihoo.browser V/ChromeTabbedActivity﹕ ⇠ onCreateWithNative [19ms]

到此为止,终于把Hugo在这个项目中用上了.

在开发和调试中的实际作用
1. 方便打log

这点毋庸置疑,一行代码解决了一个类的所有log打印的作用, 而且还自动加上了调用参数, 返回参数,执行时间等信息.

2. 性能调试更方便

方法的执行时间一目了然, 比用traceview要方便的多. 对于发现耗时方法很是有帮助.

3. 对于一些疑难bug的解决

比如我现在碰到的一个问题, 启动浏览器有时候闪屏,有时候却又不闪屏, 那很可能是启动过程不一致造成的问题。
通过脚本把所有的类前都加上@DebugLog注释
通过hugo把2次log收集起来, 再通过对比, 就可以比较容易的发现2次启动过程在执行过程中有哪些差异,找到造成bug的代码点。

4. 快速找到某个操作在代码中的对应位置

正常开发中, 不可能在所有的方法中都加上log信息, 通过hugo,例如点了某个菜单项, 可以快速定位到这个操作在项目代码中的位置,提高我们的开发效率.

碰到的一个小问题

hugo @debugLog 使用到整个类的时候, 小概率会出现行为异常.
遇到过一次, 使用到ThemeOnlinePreviewActivityV3.java上, 一直数据显示不出来. 所以在使用hugo的时候要注意这一点, 如果出现行为异常, 可以尝试把@debugLog去掉,看看是不是就正常了.

Refer:

https://github.com/JakeWharton/hugo/ //大神JakeWharton实现的一套方法调用监控.
http://www.open-open.com/lib/view/open1451870348761.html //使用介绍
https://yq.aliyun.com/articles/7104 //实现原理
==== done ====

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

推荐阅读更多精彩内容

  • Android 自定义View的各种姿势1 Activity的显示之ViewRootImpl详解 Activity...
    passiontim阅读 171,132评论 25 707
  • Spring Cloud为开发人员提供了快速构建分布式系统中一些常见模式的工具(例如配置管理,服务发现,断路器,智...
    卡卡罗2017阅读 134,510评论 18 139
  • 计划明早给女儿做早餐,今晚实验下吐司披萨
    致简繁荣阅读 149评论 0 0
  • 有位寺庙的和尚,出山远游。他来到一家客栈休息,客栈的对面的是一个大妓院,旁边是一座小诊所。他看到一群身穿华服的公子...
    海王星1984阅读 253评论 0 0
  • 先说说这个活动,是我给研究生提出的训练阅读和写作的计划。主要是输出,每天要求写200字以上并在朋友圈分享。作为老师...
    重理工杨阅读 456评论 0 0