閑人野居
          好好學習,天天向上
          posts - 57,  comments - 137,  trackbacks - 0
          ??? 關于java字節(jié)碼的處理,目前有很多工具,如bcel,asm。不過這些都需要直接跟虛擬機指令打交道。如果你不想了解虛擬機指令,可以采用javassist。javassist是jboss的一個子項目,其主要的優(yōu)點,在于簡單,而且快速。直接使用java編碼的形式,而不需要了解虛擬機指令,就能動態(tài)改變類的結構,或者動態(tài)生成類。
          ??? 下面通過一個簡單的例子,通過javassist來實現如何動態(tài)注入代碼。
          ??? 假設,存在類A,如下:
          public class A {
          ??? public void method() {
          ??????? for (int i = 0; i < 1000000; i++) {
          ??????? }
          ??????? System.out.println("method1");
          ??? }
          }
          測試類B如下:
          public class B {
          ??? public static void main(String[] args) {
          ??????? A a = new A();
          ??????? a.method();?? ?
          ??? }
          }
          現在想統(tǒng)計一下method的執(zhí)行時間,
          默認的實現是修改method:
          ?public void method() {
          ??????? long start = System.currentTimeMillis();
          ??????? for (int i = 0; i < 1000000; i++) {
          ??????? }
          ??????? System.out.println("method1");
          ??????? long end = System.currentTimeMillis();
          ??????? System.out.println(end - start);
          ??? }
          如果A的方法很多,統(tǒng)計方法的執(zhí)行時間的代碼就會相應的增加。為了減少工作量,通過動態(tài)注入代碼的形式來實現。
          修改B的main方法:
          ??? public static void main(String[] args) throws Exception {
          ?? ?? //用于取得字節(jié)碼類,必須在當前的classpath中,使用全稱
          ??????? CtClass ctClass = ClassPool.getDefault().get("org.esoft.A");
          ???????? //需要修改的方法名稱
          ??????? String mname = "method";?????? ?
          ??????? CtMethod mold = ctClass.getDeclaredMethod(mname);
          ???????? //修改原有的方法名稱
          ??????? String nname = mname + "$impl";
          ??????? mold.setName(nname);
          ???????? //創(chuàng)建新的方法,復制原來的方法
          ??????? CtMethod mnew = CtNewMethod.copy(mold, mname, ctClass, null);
          ???????? //主要的注入代碼
          ??????? StringBuffer body = new StringBuffer();
          ??????? body.append("{\nlong start = System.currentTimeMillis();\n");
          ??????? //調用原有代碼,類似于method();($$)表示所有的參數
          ??????? body.append(nname + "($$);\n");
          ??????? body.append("System.out.println(\"Call to method "
          ??????????????????? + mname
          ??????????????????? + " took \" +\n (System.currentTimeMillis()-start) + "
          ??????????????????? + "\" ms.\");\n");
          ????? ?
          ??????? body.append("}");
          ???????? //替換新方法
          ??????? mnew.setBody(body.toString());
          ???????? //增加新方法
          ??????? ctClass.addMethod(mnew);
          ??????? //類已經更改,注意不能使用A a=new A();,因為在同一個classloader中,不允許裝載同一個類兩次
          ??????? A a=(A)ctClass.toClass().newInstance();
          ??????? a.method();
          ??? }
          這只是簡單的一個應用。javassist還提供了很多的功能,用于更改類結構。有興趣的可以參考相關文檔

          posted on 2007-02-10 21:02 布衣郎 閱讀(13774) 評論(9)  編輯  收藏 所屬分類: jdk相關

          FeedBack:
          # re: 使用javassist動態(tài)注入代碼
          2007-04-19 10:06 | noname
          good!  回復  更多評論
            
          # re: 使用javassist動態(tài)注入代碼[未登錄]
          2008-04-08 19:32 | Tester
          你編譯并運行過你的代碼?   回復  更多評論
            
          # re: 使用javassist動態(tài)注入代碼[未登錄]
          2008-04-08 19:34 | Tester
          如果運行的話,

          在: A a=(A)ctClass.toClass().newInstance();上將拋出 java.lang.ClassCastException

          這是為什么呢?希望你可以解答.  回復  更多評論
            
          # re: 使用javassist動態(tài)注入代碼[未登錄]
          2008-05-12 17:25 | Tester
          http://www.csg.is.titech.ac.jp/~chiba/javassist/tutorial/tutorial.html
          上關于為什么會拋出ClassCastException講的很詳細  回復  更多評論
            
          # re: 使用javassist動態(tài)注入代碼[未登錄]
          2009-05-10 15:16 | 菜鳥
          Test.main() inserts a call to println() in the method body of say() in Hello. Then it constructs an instance of the modified Hello class and calls say() on that instance.

          Note that the program above depends on the fact that the Hello class is never loaded before toClass() is invoked. If not, the JVM would load the original Hello class before toClass() requests to load the modified Hello class. Hence loading the modified Hello class would be failed (LinkageError is thrown). For example, if main() in Test is something like this:

          public static void main(String[] args) throws Exception {
          Hello orig = new Hello();
          ClassPool cp = ClassPool.getDefault();
          CtClass cc = cp.get("Hello");
          :
          }

          then the original Hello class is loaded at the first line of main and the call to toClass() throws an exception since the class loader cannot load two different versions of the Hello class at the same time.

          If the program is running on some application server such as JBoss and Tomcat, the context class loader used by toClass() might be inappropriate. In this case, you would see an unexpected ClassCastException. To avoid this exception, you must explicitly give an appropriate class loader to toClass(). For example, if bean is your session bean object, then the following code:

          CtClass cc = ...;
          Class c = cc.toClass(bean.getClass().getClassLoader());

          would work. You should give toClass() the class loader that has loaded your program (in the above example, the class of the bean object).

          toClass() is provided for convenience. If you need more complex functionality, you should write your own class loader.

            回復  更多評論
            
          # re: 使用javassist動態(tài)注入代碼
          2009-06-12 12:04 | genle657101519
          有收獲,謝了。  回復  更多評論
            
          # re: 使用javassist動態(tài)注入代碼
          2009-08-05 19:12 | helo
          CtMethod mnew = CtNewMethod.copy(mold, mname, ctClass, null);
          應該為:CtMethod mnew = CtNewMethod.copy(mold, nname, ctClass, null);

          后面的代碼也要修改。誤人子弟,無語。。。。。  回復  更多評論
            
          # re: 使用javassist動態(tài)注入代碼[未登錄]
          2009-08-07 07:42 | joe
          @helo
          確實寫錯了  回復  更多評論
            
          # re: 使用javassist動態(tài)注入代碼
          2010-08-06 17:31 | abettor

          <2009年8月>
          2627282930311
          2345678
          9101112131415
          16171819202122
          23242526272829
          303112345

          常用鏈接

          留言簿(12)

          隨筆分類(59)

          隨筆檔案(57)

          blog

          java

          uml

          搜索

          •  

          積分與排名

          • 積分 - 357736
          • 排名 - 155

          最新評論

          閱讀排行榜

          評論排行榜

          主站蜘蛛池模板: 固安县| 武汉市| 读书| 田林县| 马尔康县| 兴国县| 长海县| 隆安县| 兖州市| 新乐市| 潜山县| 开江县| 称多县| 屏东市| 乐平市| 海阳市| 社旗县| 云阳县| 丰都县| 松滋市| 沅陵县| 龙山县| 井研县| 绍兴县| 馆陶县| 通渭县| 邻水| 义乌市| 清苑县| 瓮安县| 偃师市| 桦川县| 剑川县| 义乌市| 略阳县| 延津县| 新平| 凌海市| 祁阳县| 望城县| 新源县|