BTrace施用總結

來源:互聯網
上載者:User

http://www.myexception.cn/program/729263.html

BTrace使用總結
一、背景
       在生產環境中可能經常遇到各種問題,定位問題需要擷取程式運行時的資料資訊,如方法參數、傳回值、全域變數、堆棧資訊等。為了擷取這些資料資訊,我們可以通過改寫代碼,增加日誌資訊的列印,再發布到生產環境。通過這種方式,一方面將增大定位問題的成本和周期,對於緊急問題無法做到及時響應;另一方面重新部署後環境可能已被破壞,很難重新問題的情境。

二、BTrace功能
       BTrace天生就為解決這類問題而來,它可以動態地跟蹤java運行程式。通過hotswap技術,動態將跟蹤位元組碼注入到運行類中,對運行代碼侵入較小,對效能上的影響可以忽略不計。
       BTrace在使用上有很多限制條件,如不能建立對象、數組、拋出和捕獲異常、迴圈等,具體限制條件參考使用者文檔中的BTrace Restrictions。使用者文檔地址: http://kenai.com/projects/btrace/pages/UserGuide。
       根據官方聲明,不當地使用btrace可能導致jvm崩潰,如BTrace使用錯誤的.class檔案,Hotspot JVM自身存在的hotswap bug等。可以先在本地驗證BTrace指令碼的正確性,再傳到生產環境中定位問題。

    
三、安裝步驟
1. 下載安裝壓縮包,最新版本的是1.2.1,: http://kenai.com/projects/btrace/downloads/directory/releases。
2. 解壓縮,命令指令碼放在bin目錄中。
3. 設定指令碼環境變數。
4. 增加指令碼可執行許可權。

四、使用方法
BTrace主要包含btracec和btrace兩個命令編譯和啟動BTrace指令碼:
1. btrace
功能: 用於運行BTrace跟蹤程式。 
命令格式:
btrace [-I <include-path>] [-p <port>] [-cp <classpath>] <pid> <btrace-script> [<args>]
樣本: 
btrace -cp build/  1200 AllCalls1.java 
參數含義:
include-path指定標頭檔的路徑,用於指令碼預先處理功能,可選;
port指定BTrace agent的服務端監聽連接埠號碼,用來監聽clients,預設為2020,可選;
classpath用來指定類載入路徑,預設為當前路徑,可選;
pid表示進程號,可通過jps命令擷取; 
btrace-script即為BTrace指令碼;btrace指令碼如果以.java結尾,會先編譯再提交執行。可使用btracec命令對指令碼進行先行編譯。
args是BTrace指令碼選擇性參數,在指令碼中可通過"$"和"$length"擷取參數資訊。

2. btracec
功能: 用於先行編譯BTrace指令碼,用於在編譯時間期驗證指令碼正確性。
btracec [-I <include-path>] [-cp <classpath>] [-d <directory>] <one-or-more-BTrace-.java-files>
參數意義同btrace命令一致,directory表示編譯結果輸出目錄。

3. btracer
功能: btracer命令同時啟動應用程式和BTrace指令碼,即在應用程式啟動過程中使用BTrace指令碼。而btrace命令針對已運行程式執行BTrace指令碼。
命令格式: 
btracer <pre-compiled-btrace.class> <application-main-class> <application-args>
參數說明: 
pre-compiled-btrace.class表示經過btracec編譯後的BTrace指令碼。
application-main-class表示應用程式代碼;
application-args表示應用程式參數。
該命令的等價寫法為:
java -javaagent:btrace-agent.jar=script=<pre-compiled-btrace-script1>[,<pre-compiled-btrace-script1>]* <MainClass> <AppArguments>

4. jvisualvm外掛程式
BTrace提供了jvisualvm外掛程式,強烈推薦在jvisualvm中編寫和測試BTrace指令碼,啟動、關閉、發送事件、增加classpath都非常方便。


五、BTrace實戰
1. 範例程式碼
範例程式碼定義了Counter計數器,有一個add()方法,每次增加隨機值,總數儲存在totalCount屬性中。

package com.learnworld;import java.util.Random;public class BTraceTest {public static void main(String[] args) throws Exception {Random random = new Random();// 計數器Counter counter = new Counter();while (true) {// 每次增加隨機值counter.add(random.nextInt(10));Thread.sleep(1000);}}}


package com.learnworld;public class Counter {// 總數private static int totalCount = 0;public int add(int num) throws Exception {totalCount += num;sleep();return totalCount;}public void sleep() throws Exception {Thread.sleep(1000);}}




2. 常見使用情境
下面通過幾個常見使用情境示範如何使用BTrace指令碼。

1) 擷取add()方法參數值和傳回值。

import com.sun.btrace.annotations.*;import static com.sun.btrace.BTraceUtils.*;@BTracepublic class TracingScript {   @OnMethod(      clazz="com.learnworld.Counter",      method="add",      location=@Location(Kind.RETURN)   )   public static void traceExecute(int num,@Return int result){     println("====== ");     println(strcat("parameter num: ",str(num)));     println(strcat("return value:",str(result)));   }}



2) 定時擷取Counter類的屬性值totalCount。

import com.sun.btrace.annotations.*;import static com.sun.btrace.BTraceUtils.*;@BTracepublic class TracingScript {   private static Object totalCount = 0;      @OnMethod(      clazz="com.learnworld.Counter",      method="add",      location=@Location(Kind.RETURN)   )    public static void traceExecute(@Self com.learnworld.Counter counter){     totalCount = get(field("com.learnworld.Counter","totalCount"), counter);   }        @OnTimer(1000)   public static void print(){     println("====== ");     println(strcat("totalCount: ",str(totalCount)));   }}




3) 擷取add方法執行時間。

import com.sun.btrace.annotations.*;import static com.sun.btrace.BTraceUtils.*;@BTracepublic class TracingScript {   @TLS private static long startTime = 0;      @OnMethod(      clazz="com.learnworld.Counter",      method="add"   )    public static void startExecute(){     startTime = timeNanos();   }        @OnMethod(      clazz="com.learnworld.Counter",      method="add",      location=@Location(Kind.RETURN)   )    public static void endExecute(@Duration long duration){     long time = timeNanos() - startTime;     println(strcat("execute time(nanos): ", str(time)));     println(strcat("duration(nanos): ", str(duration)));   } }



4) 擷取add()方法調用方法sleep()次數。

import com.sun.btrace.annotations.*;import static com.sun.btrace.BTraceUtils.*;@BTracepublic class TracingScript {   private static long count;         @OnMethod(      clazz="/.*/",      method="add",      location=@Location(value=Kind.CALL, clazz="/.*/", method="sleep")   )   public static void traceExecute(@ProbeClassName String pcm, @ProbeMethodName String pmn,   @TargetInstance Object instance,  @TargetMethodOrField String method){     println("====== ");     println(strcat("ProbeClassName: ",pcm));     println(strcat("ProbeMethodName: ",pmn));     println(strcat("TargetInstance: ",str(classOf(instance))));     println(strcat("TargetMethodOrField : ",str(method)));     count++;   }      @OnEvent   public static void getCount(){       println(strcat("count: ", str(count)));   }}



六、參考文檔
1. userGuide: http://kenai.com/projects/btrace/pages/UserGuide 
2. JAVA doc:  http://btrace.kenai.com/javadoc/1.2/index.html
3. BTrace使用者手冊<譯>,http://macrochen.iteye.com/blog/838920 
4. btrace使用簡介,http://rdc.taobao.com/team/jm/archives/509
5. btrace記憶,http://agapple.iteye.com/blog/962119
6. btrace一些你不知道的事(源碼入手),http://agapple.iteye.com/blog/1005918

聯繫我們

該頁面正文內容均來源於網絡整理,並不代表阿里雲官方的觀點,該頁面所提到的產品和服務也與阿里云無關,如果該頁面內容對您造成了困擾,歡迎寫郵件給我們,收到郵件我們將在5個工作日內處理。

如果您發現本社區中有涉嫌抄襲的內容,歡迎發送郵件至: info-contact@alibabacloud.com 進行舉報並提供相關證據,工作人員會在 5 個工作天內聯絡您,一經查實,本站將立刻刪除涉嫌侵權內容。

A Free Trial That Lets You Build Big!

Start building with 50+ products and up to 12 months usage for Elastic Compute Service

  • Sales Support

    1 on 1 presale consultation

  • After-Sales Support

    24/7 Technical Support 6 Free Tickets per Quarter Faster Response

  • Alibaba Cloud offers highly flexible support services tailored to meet your exact needs.