BTrace 使用總結

Coding-lover發表於2015-10-07

一、背景

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

二、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屬性中。
Btracetest.java程式碼

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);  
        }  
    }  
} 

Counter.java程式碼

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.*;  

@BTrace  
public 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.*;  

@BTrace  
public 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.*;  

@BTrace  
public 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.*;  

@BTrace  
public 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

轉載自:BTrace使用總結