閑人野居
          好好學習,天天向上
          posts - 57,  comments - 137,  trackbacks - 0
          ??? 關于java字節碼的處理,目前有很多工具,如bcel,asm。不過這些都需要直接跟虛擬機指令打交道。如果你不想了解虛擬機指令,可以采用javassist。javassist是jboss的一個子項目,其主要的優點,在于簡單,而且快速。直接使用java編碼的形式,而不需要了解虛擬機指令,就能動態改變類的結構,或者動態生成類。
          ??? 下面通過一個簡單的例子,通過javassist來實現如何動態注入代碼。
          ??? 假設,存在類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();?? ?
          ??? }
          }
          現在想統計一下method的執行時間,
          默認的實現是修改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的方法很多,統計方法的執行時間的代碼就會相應的增加。為了減少工作量,通過動態注入代碼的形式來實現。
          修改B的main方法:
          ??? public static void main(String[] args) throws Exception {
          ?? ?? //用于取得字節碼類,必須在當前的classpath中,使用全稱
          ??????? CtClass ctClass = ClassPool.getDefault().get("org.esoft.A");
          ???????? //需要修改的方法名稱
          ??????? String mname = "method";?????? ?
          ??????? CtMethod mold = ctClass.getDeclaredMethod(mname);
          ???????? //修改原有的方法名稱
          ??????? String nname = mname + "$impl";
          ??????? mold.setName(nname);
          ???????? //創建新的方法,復制原來的方法
          ??????? 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 布衣郎 閱讀(13773) 評論(9)  編輯  收藏 所屬分類: jdk相關

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

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

          這是為什么呢?希望你可以解答.  回復  更多評論
            
          # re: 使用javassist動態注入代碼[未登錄]
          2008-05-12 17:25 | Tester
          http://www.csg.is.titech.ac.jp/~chiba/javassist/tutorial/tutorial.html
          上關于為什么會拋出ClassCastException講的很詳細  回復  更多評論
            
          # re: 使用javassist動態注入代碼[未登錄]
          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動態注入代碼
          2009-06-12 12:04 | genle657101519
          有收獲,謝了。  回復  更多評論
            
          # re: 使用javassist動態注入代碼
          2009-08-05 19:12 | helo
          CtMethod mnew = CtNewMethod.copy(mold, mname, ctClass, null);
          應該為:CtMethod mnew = CtNewMethod.copy(mold, nname, ctClass, null);

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

          <2007年2月>
          28293031123
          45678910
          11121314151617
          18192021222324
          25262728123
          45678910

          常用鏈接

          留言簿(12)

          隨筆分類(59)

          隨筆檔案(57)

          blog

          java

          uml

          搜索

          •  

          積分與排名

          • 積分 - 357711
          • 排名 - 155

          最新評論

          閱讀排行榜

          評論排行榜

          主站蜘蛛池模板: 霍州市| 揭西县| 敦煌市| 景洪市| 新巴尔虎左旗| 汨罗市| 怀安县| 平潭县| 都匀市| 大庆市| 泰来县| 章丘市| 泰宁县| 响水县| 都匀市| 桐柏县| 临安市| 融水| 伊宁市| 柏乡县| 肃宁县| 屯昌县| 宜黄县| 应用必备| 青川县| 安丘市| 玉树县| 沙坪坝区| 西青区| 兴化市| 县级市| 榆树市| 中方县| 巨鹿县| 岱山县| 余干县| 乐陵市| 涟源市| 桃源县| 泸溪县| 宁陵县|