青青草原综合久久大伊人导航_色综合久久天天综合_日日噜噜夜夜狠狠久久丁香五月_热久久这里只有精品

隨筆-30  評論-67  文章-0  trackbacks-0

最近看了Java 編程的動態(tài)性,第 7 部分: 用 BCEL 設(shè)計字節(jié)碼,網(wǎng)址是http://www-128.ibm.com/developerworks/cn/java/j-dyn0414/

其中的示范代碼解釋的不是很詳細,這方面的中文資料又少,只好自己花時間看下去。bcel的類庫卻是不夠友好,api文檔也是走馬觀花的點一下,很多函數(shù)沒有說明。理解基本靠猜,還好有個示范代碼。下面對自己理解的東西做個記錄,詳細的代碼可以到上面的鏈接下載。


import java.io.FileOutputStream;
import java.io.IOException;

import org.apache.bcel.Constants;
import org.apache.bcel.classfile.ClassParser;
import org.apache.bcel.classfile.JavaClass;
import org.apache.bcel.classfile.Method;
import org.apache.bcel.generic.ClassGen;
import org.apache.bcel.generic.ConstantPoolGen;
import org.apache.bcel.generic.InstructionConstants;
import org.apache.bcel.generic.InstructionFactory;
import org.apache.bcel.generic.InstructionList;
import org.apache.bcel.generic.MethodGen;
import org.apache.bcel.generic.ObjectType;
import org.apache.bcel.generic.PUSH;
import org.apache.bcel.generic.Type;

public class BCELTiming {
??? /**
???? * Add timing wrapper to method of class. The method can accept any
???? * arguments and return any type (including void), but must be a normal
???? * (non-static, non-initializer) method to be used with this code as
???? * currently implemented. Handling the other types of methods would not
???? * involve any fundamental changes to the code.
???? *
???? * @param cgen
???? *??????????? generator for class being modified
???? * @param method
???? *??????????? current method to be enhanced with timing wrapper
???? */

??? private static void addWrapper(ClassGen cgen, Method method) {

??????? // set up the construction tools
??????? InstructionFactory ifact = new InstructionFactory(cgen);
??????? InstructionList ilist = new InstructionList();
??????? ConstantPoolGen pgen = cgen.getConstantPool();
??????? String cname = cgen.getClassName();
??????? MethodGen wrapgen = new MethodGen(method, cname, pgen);
??????? wrapgen.setInstructionList(ilist);

??????? // rename a copy of the original method
??????? MethodGen methgen = new MethodGen(method, cname, pgen);
??????? cgen.removeMethod(method);
??????? String iname = methgen.getName() + "$impl";
??????? methgen.setName(iname);
??????? cgen.addMethod(methgen.getMethod());
??????? //以上是一下初始化的工作
???????

??????? // compute the size of the calling parameters
??????? // operand stack操作數(shù)堆棧
??????? Type[] types = methgen.getArgumentTypes(); // 取出參數(shù)類型數(shù)組

??????? // solt代表本地變量的堆棧偏移量,里頭儲存了調(diào)用methen代表的函數(shù)的參數(shù)
??????? int slot = methgen.isStatic() ? 0 : 1; // 這種方式與Java如何處理方法調(diào)用有關(guān)。對于非靜態(tài)的方法,每次調(diào)用的第一個(隱藏的)參數(shù)是目標(biāo)對象的this引用(就是位置0儲存的內(nèi)容)。
??????? for (int i = 0; i < types.length; i++) {
??????????? slot += types[i].getSize();// 累計個個參數(shù)類型的長度,
??????? }
??????? // 現(xiàn)在solt指向最后一個參數(shù)的下一個位置

??????? // save time prior to invocation
??????? // 調(diào)用靜態(tài)的long java.lang.System.currentTimeMillis()方法,調(diào)用結(jié)束后函數(shù)的返回的long類型的值會壓入operand stack操作數(shù)堆棧
??????? ilist.append(ifact.createInvoke("java.lang.System",
??????????????? "currentTimeMillis", Type.LONG, Type.NO_ARGS,
??????????????? Constants.INVOKESTATIC));
??????? ilist.append(InstructionFactory.createStore(Type.LONG, slot));// 將operand stack的top保存到本地變量堆棧的slot位置,operand stack彈出long值

??????? // call the wrapped method
??????? int offset = 0; // 偏移量
??????? short invoke = Constants.INVOKESTATIC; // 預(yù)先設(shè)置為調(diào)用靜態(tài)函數(shù)
??????? if (!methgen.isStatic()) { // 如果不是調(diào)用靜態(tài)函數(shù),將調(diào)用的第一個(隱藏的)參數(shù)(目標(biāo)對象的this引用)壓入operand stack
??????????? ilist.append(InstructionFactory.createLoad(Type.OBJECT, 0));
??????????? offset = 1;// 偏移量加1
??????????? invoke = Constants.INVOKEVIRTUAL;// 設(shè)置為調(diào)用非靜態(tài)函數(shù)
??????? }
??????? for (int i = 0; i < types.length; i++) { // 遍歷所有參數(shù)
??????????? Type type = types[i];
??????????? ilist.append(InstructionFactory.createLoad(type, offset)); // 按參數(shù)類型把參數(shù)一個個從本地變量堆棧取出,壓入operand stack
??????????? offset += type.getSize();
??????? }
??????? Type result = methgen.getReturnType();// 取得要調(diào)用函數(shù)的返回值類型
??????? ilist.append(ifact.createInvoke(cname, iname, result, types, invoke));// 調(diào)用方法名為iname的函數(shù)

??????? // store result for return later
??????? if (result != Type.VOID) {
??????????? ilist.append(InstructionFactory.createStore(result, slot + 2)); // 將名為iname的函數(shù)返回值復(fù)制到本地變量堆棧的slot+2的位置上
??????? }

??????? // print time required for method call
??????? // 獲取靜態(tài)對象java.lang.System.out的引用,返回值壓入operand stack
??????? ilist.append(ifact.createFieldAccess("java.lang.System", "out",
??????????????? new ObjectType("java.io.PrintStream"), Constants.GETSTATIC));
??????? ilist.append(InstructionConstants.DUP);// 取operand stack的top,壓入operand stack。完成后load_stack的頭兩個元素是靜態(tài)對象java.lang.System.out的引用
??????? ilist.append(InstructionConstants.DUP);// 取operand stack的top,壓入operand stack。現(xiàn)在有3個java.lang.System.out的引用。供下面3次調(diào)用out.print()函數(shù)使用
??????? String text = "Call to method " + methgen.getName() + " took ";
??????? ilist.append(new PUSH(pgen, text));// 將text放入pgen(代表常量池),并把其在pgen的引用壓入operand stack(供out.print(Sting)調(diào)用的參數(shù))
??????? ilist.append(ifact.createInvoke("java.io.PrintStream", "print",
??????????????????????? Type.VOID, new Type[] { Type.STRING },
??????????????????????? Constants.INVOKEVIRTUAL));// 調(diào)用結(jié)束,operand stack彈出一個String的引用和一個out的引用(還剩2個out),函數(shù)沒有返回值

??????? ilist.append(ifact.createInvoke("java.lang.System",
??????????????? "currentTimeMillis", Type.LONG, Type.NO_ARGS,
??????????????? Constants.INVOKESTATIC));// 調(diào)用java.lang.System.currentTimeMillis()方法,調(diào)用結(jié)束后函數(shù)的返回的long類型的值會壓入堆棧operand stack
??????? ilist.append(InstructionFactory.createLoad(Type.LONG, slot));// 從本地變量堆棧的slot位置載入先前儲存的long值,壓入operand stack
??????? ilist.append(InstructionConstants.LSUB);// 調(diào)用long的減法指令,彈出2個long值,并把結(jié)果壓入operand stack,現(xiàn)在operand stack的top第一個是long,第二個是out的引用
??????? ilist.append(ifact.createInvoke("java.io.PrintStream", "print",
??????????????? Type.VOID, new Type[] { Type.LONG }, Constants.INVOKEVIRTUAL));// 調(diào)用out.print(long)方法
??????? ilist.append(new PUSH(pgen, " ms."));// 將String對象" ms."放入pgen,并把其在pgen的引用壓入operand stack(供out.print(Sting)調(diào)用的參數(shù))
??????? ilist
??????????????? .append(ifact.createInvoke("java.io.PrintStream", "println",
??????????????????????? Type.VOID, new Type[] { Type.STRING },
??????????????????????? Constants.INVOKEVIRTUAL));

??????? // return result from wrapped method call
??????? if (result != Type.VOID) {
??????????? ilist.append(InstructionFactory.createLoad(result, slot + 2));// 處理返回值,如果不為空,從本地對象堆棧的slot+2位置讀取指定類型的返回值壓入operand stack
??????? }
??????? ilist.append(InstructionFactory.createReturn(result)); //調(diào)用處理返回值的指令,result為返回值的類型

??????? //下面是一下掃尾工作
??????? // finalize the constructed method
??????? wrapgen.stripAttributes(true);
??????? wrapgen.setMaxStack();
??????? wrapgen.setMaxLocals();
??????? cgen.addMethod(wrapgen.getMethod());
??????? ilist.dispose();
??? }

??? public static void main(String[] argv) {
??????? if (argv.length == 2 && argv[0].endsWith(".class")) {
??????????? try {

??????????????? JavaClass jclas = new ClassParser(argv[0]).parse();
??????????????? ClassGen cgen = new ClassGen(jclas);
??????????????? Method[] methods = jclas.getMethods();
??????????????? int index;
??????????????? for (index = 0; index < methods.length; index++) {
??????????????????? if (methods[index].getName().equals(argv[1])) {
??????????????????????? break;
??????????????????? }
??????????????? }
??????????????? if (index < methods.length) {
??????????????????? addWrapper(cgen, methods[index]);
??????????????????? FileOutputStream fos = new FileOutputStream(argv[0]);
??????????????????? cgen.getJavaClass().dump(fos);
??????????????????? fos.close();
??????????????? } else {
??????????????????? System.err.println("Method " + argv[1] + " not found in "
??????????????????????????? + argv[0]);
??????????????? }
??????????? } catch (IOException ex) {
??????????????? ex.printStackTrace(System.err);
??????????? }

??????? } else {
??????????? System.out.println("Usage: BCELTiming class-file method-name");
??????? }
??? }
}


相對javassist,bcel確實比較復(fù)雜。但是bcel給我的感覺比較自由,有種一切盡在掌握的感覺,這一點比較我喜歡。
雖然自由,但是用bcel寫一個類的實現(xiàn)絕對不會是一件讓人開心的事情,如果碰巧你的實現(xiàn)又稍微復(fù)雜了點(比如實現(xiàn)了xx接口,又增加了一些函數(shù)和成員變量)。而且你的實現(xiàn)未必會比javac編譯出來的代碼效率高(不考慮jvm的動態(tài)優(yōu)化)。
不過,先把你要實現(xiàn)的類的代碼寫出來,按照代碼來寫bcel的實現(xiàn),會降低些實現(xiàn)難度。

posted on 2006-05-31 16:44 含笑半步癲 閱讀(2501) 評論(4)  編輯 收藏 引用 所屬分類: java

評論:
# re: 學(xué)用BCEL設(shè)計字節(jié)碼的記錄(java) 2006-08-31 09:41 | lai
我嘗試了編譯這個程序,可是沒有輸出啊!估計是因為在StringBuilder中沒有傳進參數(shù)的原因,請問在eclipse中應(yīng)該怎么做呢
  回復(fù)  更多評論
  
# re: 學(xué)用BCEL設(shè)計字節(jié)碼的記錄(java) 2006-09-07 22:47 | 含笑半步癲
main的函數(shù)是需要參數(shù)的。具體如何運行請參考上面給出的ibm的文章  回復(fù)  更多評論
  
# re: 學(xué)用BCEL設(shè)計字節(jié)碼的記錄(java) 2008-01-04 17:52 | allen
這個代碼好像是丹尼斯寫的 IBM網(wǎng)站Java動態(tài)編程系列文章一共8篇  回復(fù)  更多評論
  
# re: 學(xué)用BCEL設(shè)計字節(jié)碼的記錄(java) 2010-05-20 15:16 | ewwerpm
包org.apache.bcel.util.BCELifier和com.sun.org.apache.bcel.internal.util.BCELifier有什么區(qū)別?com.sun.org.apache.bcel.internal.util.BCELifier是裝 JDK1.6的時候JRE里面自帶的,org.apache.bcel.util.BCELifier需要從網(wǎng)上下載,類名字是一樣的,不知道他們在使用的時候有什么區(qū)別?我只能用org.apache.bcel.util.BCELifier包將HelloWorld.class文件反編譯成HelloWorldCreator.java文件,但是在執(zhí)行 java HelloWorldCrator 的時候沒用設(shè)想中的輸出啊。
public class HelloWorld{
public static void main(String args[]){
System.err.println("Hello World through BCEL!");
}
}  回復(fù)  更多評論
  
青青草原综合久久大伊人导航_色综合久久天天综合_日日噜噜夜夜狠狠久久丁香五月_热久久这里只有精品
  • <ins id="pjuwb"></ins>
    <blockquote id="pjuwb"><pre id="pjuwb"></pre></blockquote>
    <noscript id="pjuwb"></noscript>
          <sup id="pjuwb"><pre id="pjuwb"></pre></sup>
            <dd id="pjuwb"></dd>
            <abbr id="pjuwb"></abbr>
            永久免费视频成人| 欧美日韩视频一区二区| 亚洲国产天堂久久综合网| 久久激情久久| 久久久91精品国产一区二区三区| 亚洲一区二区在| 午夜精品成人在线视频| 久久精品亚洲一区二区| 麻豆精品视频在线| 狠狠色丁香婷婷综合| 国产一区二区三区久久精品| 激情一区二区三区| 亚洲黄色尤物视频| 一本到12不卡视频在线dvd| 亚洲女人天堂成人av在线| 久久精视频免费在线久久完整在线看| 久久久视频精品| 亚洲黄色成人久久久| 亚洲一区二区久久| 麻豆精品国产91久久久久久| 欧美日韩一卡| 精品1区2区3区4区| 亚洲午夜伦理| 狂野欧美一区| 亚洲图片在线观看| 美国十次了思思久久精品导航| 欧美日韩午夜在线| 狠狠综合久久| 欧美亚洲一级片| 欧美韩日一区| 午夜精品久久一牛影视| 欧美日韩国产精品自在自线| 国产一区久久久| 亚洲一区免费网站| 亚洲国产精品va在线看黑人动漫 | 亚洲一级黄色片| 久久天天狠狠| 亚洲午夜激情在线| 欧美黄色aa电影| 在线免费观看一区二区三区| 欧美一区二区免费| 一区二区成人精品 | 国产亚洲精品一区二555| 亚洲免费观看高清在线观看| 久久久亚洲欧洲日产国码αv| 中文高清一区| 欧美日韩国产色综合一二三四| 激情综合电影网| 久久大逼视频| 亚洲香蕉伊综合在人在线视看| 蜜臀久久99精品久久久久久9| 国产一区二区精品| 欧美一区二区视频97| 中文欧美日韩| 国产精品久久久久久久久婷婷 | 亚洲第一页中文字幕| 久久精品免费| 红桃视频国产精品| 久久久久久一区二区三区| 久久久久久**毛片大全| 亚洲制服欧美中文字幕中文字幕| 欧美日韩精品二区| 一本不卡影院| av不卡在线看| 国产精品久久久一本精品| 国产精品99久久久久久久女警| 亚洲国产欧美一区二区三区久久 | 亚洲一区免费网站| 在线亚洲免费| 国产麻豆精品视频| 久久久精品国产免费观看同学| 亚洲宅男天堂在线观看无病毒| 国产精品欧美日韩一区| 久久精品五月婷婷| 久久伊人亚洲| 亚洲精品一区二区三区99| 亚洲欧洲视频| 国产精品久久久久久久久免费桃花 | 一区二区日韩| 亚洲神马久久| 国产三级精品在线不卡| 男女激情视频一区| 欧美精品入口| 先锋影院在线亚洲| 久久久久久久久久久久久女国产乱 | 国产精品欧美日韩一区二区| 久久精品国产77777蜜臀| 久久久视频精品| 日韩视频永久免费观看| 亚洲欧美国产va在线影院| 一色屋精品视频免费看| 亚洲精品乱码久久久久久| 国产精品一区久久久久| 欧美va亚洲va国产综合| 欧美日在线观看| 久久经典综合| 欧美日韩美女在线| 久久综合九色欧美综合狠狠| 欧美成人精品1314www| 午夜国产精品视频| 蜜桃精品一区二区三区| 欧美一区二区高清在线观看| 欧美成人精品不卡视频在线观看| 亚洲欧美日韩国产成人| 欧美阿v一级看视频| 久久9热精品视频| 欧美日韩免费区域视频在线观看| 久久综合五月天婷婷伊人| 欧美日韩国产一中文字不卡 | 久久久久久久高潮| 欧美成人精品在线播放| 欧美一区二区三区视频在线观看| 免费视频亚洲| 欧美亚洲专区| 欧美午夜精品久久久久久久| 亚洲福利免费| 国产一区二区三区观看| 99精品国产一区二区青青牛奶| 在线不卡中文字幕| 欧美在线国产| 欧美在线一二三| 欧美午夜在线一二页| 欧美激情亚洲激情| 尤物yw午夜国产精品视频| 午夜亚洲性色视频| 午夜在线视频一区二区区别| 欧美日韩在线播放三区四区| 亚洲经典三级| 亚洲欧洲综合| 老司机一区二区三区| 美乳少妇欧美精品| 在线观看亚洲a| 久久躁狠狠躁夜夜爽| 开心色5月久久精品| 国内成人精品2018免费看| 欧美一区二区视频在线| 久久久久久久久久久一区| 国产亚洲综合精品| 欧美中文字幕在线| 久久男人资源视频| 伊人色综合久久天天| 久久午夜羞羞影院免费观看| 米奇777超碰欧美日韩亚洲| 激情六月婷婷久久| 久久免费视频网| 欧美福利电影网| 亚洲免费观看高清在线观看| 欧美片第1页综合| 在线一区亚洲| 久久久久99| 亚洲高清av在线| 欧美经典一区二区三区| 一区二区三区欧美日韩| 午夜精品亚洲| 狠狠色综合色区| 欧美不卡高清| 中文有码久久| 久久综合五月天婷婷伊人| 亚洲激情中文1区| 国产精品v日韩精品| 午夜老司机精品| 欧美成人国产va精品日本一级| 日韩视频一区二区三区在线播放 | 国产精品大全| 亚洲欧美中文字幕| 欧美国产丝袜视频| 亚洲一区影院| 亚洲第一网站免费视频| 欧美午夜激情在线| 久久久久久亚洲精品中文字幕| 欧美福利视频| 亚洲欧美偷拍卡通变态| 亚洲国产欧美日韩| 欧美午夜久久| 欧美综合国产精品久久丁香| 最新亚洲一区| 久久成人综合网| 一区二区精品国产| 精品51国产黑色丝袜高跟鞋| 亚洲激情一区二区| 欧美一区二区免费观在线| 国产视频精品网| 欧美日韩不卡一区| 久久久欧美精品sm网站| 亚洲小视频在线| 最新中文字幕亚洲| 久久婷婷麻豆| 亚洲欧美一区二区视频| 亚洲精品欧美极品| 国模叶桐国产精品一区| 国产精品劲爆视频| 欧美电影资源| 久久亚洲精品一区二区| 欧美一区二区三区四区高清| 夜夜嗨av一区二区三区四季av | 亚洲欧洲av一区二区三区久久| 亚洲精品乱码| 亚洲国产精品成人精品| 美女福利精品视频| 久久久精品免费视频|