<rt id="bn8ez"></rt>
<label id="bn8ez"></label>

  • <span id="bn8ez"></span>

    <label id="bn8ez"><meter id="bn8ez"></meter></label>

    閑人野居
    好好學習,天天向上
    posts - 57,  comments - 137,  trackbacks - 0
    ??? 關于java字節(jié)碼的處理,目前有很多工具,如bcel,asm。不過這些都需要直接跟虛擬機指令打交道。如果你不想了解虛擬機指令,可以采用javassist。javassist是jboss的一個子項目,其主要的優(yōu)點,在于簡單,而且快速。直接使用java編碼的形式,而不需要了解虛擬機指令,就能動態(tài)改變類的結構,或者動態(tài)生成類。
    ??? 下面通過一個簡單的例子,通過javassist來實現(xiàn)如何動態(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();?? ?
    ??? }
    }
    現(xiàn)在想統(tǒng)計一下method的執(zhí)行時間,
    默認的實現(xiàn)是修改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)注入代碼的形式來實現(xiàn)。
    修改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();($$)表示所有的參數(shù)
    ??????? 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 布衣郎 閱讀(13767) 評論(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年5月>
    262728293012
    3456789
    10111213141516
    17181920212223
    24252627282930
    31123456

    常用鏈接

    留言簿(12)

    隨筆分類(59)

    隨筆檔案(57)

    blog

    java

    uml

    搜索

    •  

    積分與排名

    • 積分 - 357266
    • 排名 - 155

    最新評論

    閱讀排行榜

    評論排行榜

    主站蜘蛛池模板: 91精品免费在线观看| 无码免费一区二区三区免费播放| 免费影院未满十八勿进网站| 亚洲国产综合专区电影在线| 久久99免费视频| 亚洲色图国产精品| 国产成人精品免费视频动漫 | 亚洲av成人一区二区三区| 最近2019中文字幕免费大全5| 久久亚洲精品无码VA大香大香| 91av在线免费视频| 亚洲天堂2016| 日本不卡在线观看免费v| 国产精品亚洲专区在线播放 | 久久精品国产亚洲网站| 无码日韩精品一区二区三区免费 | 亚洲av极品无码专区在线观看| 中字幕视频在线永久在线观看免费| 亚洲伦理中文字幕| 四虎影永久在线高清免费| 日韩一区二区三区免费播放| 亚洲女初尝黑人巨高清| 久久久久久国产精品免费无码| 日韩亚洲产在线观看| 国产精品嫩草影院免费| 国产精品一区二区三区免费| 久久精品国产亚洲AV高清热| 国语成本人片免费av无码| 免费精品视频在线| 亚洲毛片在线观看| 午夜毛片不卡免费观看视频| 亚欧乱色国产精品免费视频| 亚洲制服中文字幕第一区| 免费看的一级毛片| 久久久精品国产亚洲成人满18免费网站 | 99久久99久久精品免费观看| 亚洲伊人久久大香线蕉结合| 亚洲成A人片在线观看无码3D| 99视频在线免费看| 边摸边吃奶边做爽免费视频网站| 亚洲AV无码精品色午夜果冻不卡 |