免费视频淫片aa毛片_日韩高清在线亚洲专区vr_日韩大片免费观看视频播放_亚洲欧美国产精品完整版

打開APP
userphoto
未登錄

開通VIP,暢享免費電子書等14項超值服

開通VIP
java通過修改類的字節(jié)碼實現(xiàn)aop功能
      在jdk5.0版本中,添加了一個新的接口java.lang.instrument.Instrumentation,這個接口可以用來在類加載的時候對類的字節(jié)碼進行修改,改變類的功能的實現(xiàn)。通過這種方法,我們就可以實現(xiàn)aop的功能,這是和proxy動態(tài)代理的不同的另一種aop的實現(xiàn)。
     這種方法的原理很簡單,就是通過修改編譯器已經(jīng)編譯好的字節(jié)碼來修改類,和我們通常修改類源碼來實現(xiàn)對類的修改差不多。只不過是在兩個不同的層次進行的修改。
     一下是使用這個接口的具體方法。
    第一步:
    需要實現(xiàn)  java.lang.instrument.Instrumentation這個接口,通過在這個接口的實現(xiàn)類中,在加載類的時候修改類的字節(jié)碼。以下是接口的實現(xiàn) Transformer.java
public class Transformer implements ClassFileTransformer {
   //在程序運行之前就被jvm調用的方法
   public static void premain(String args, Instrumentation inst) {
        inst.addTransformer(new Transformer());
    }
    public byte[] transform(ClassLoader loader, String className,
            Class<?> classBeingRedefined, ProtectionDomain protectionDomain,
            byte[] classfileBuffer) throws IllegalClassFormatException {
        byte[] result =null; //定義新的字節(jié)碼存儲變量
        //在此通過修改類的字節(jié)碼 classfileBuffer,來更改類。
        return result; //返回新的字節(jié)碼
    }
}
第二步:
需要將
Transformer類的class文件生成jar文件,如t.jar,在jar文件的MANIFEST.MF添加 Premain-Class: Transformer 這一項。在運行程序時,需要在jvm使用如下參數(shù) -javaagent:t.jar 。這樣jvm在運行時,就會先找到Premain-Class: Transformer  這一項指定的Transformer類的premain方法,運行此方法。也就實現(xiàn)了在以后的類的加載中,對類的字節(jié)碼進行修改的功能。

        就是使用
java.lang.instrument.Instrumentation接口的方法。那么如何實現(xiàn)aop功能呢,這就需要在更改字節(jié)碼的時候,在類的功能上添加aop方面的功能。
       現(xiàn)在我們看看如何實現(xiàn)在方法的調用前和返回時打印當前時間的aop功能。
      先更改Transformer的transform方法
public byte[] transform(ClassLoader loader, String className,
            Class<?> classBeingRedefined, ProtectionDomain protectionDomain,
            byte[] classfileBuffer) throws IllegalClassFormatException {
        byte[] result =classfileBuffer; //定義新的字節(jié)碼存儲變量
        //在此通過修改類的字節(jié)碼 classfileBuffer,來更改類。
        if (loader != ClassLoader.getSystemClassLoader()) {
            return classfileBuffer;
        }

        ClassReader reader = new ClassReader(classfileBuffer);
        ClassWriter writer = new ClassWriter(true);
        ClassAdapter adapter = new PerfClassAdapter(writer, className);
        reader.accept(adapter, true);
        result = writer.toByteArray();
        return result; //返回新的字節(jié)碼
    }

這樣就通過PerfClassAdapter類來修改字節(jié)碼.
PerfClassAdapter.java
public class PerfClassAdapter extends ClassAdapter {
    private String className;
    public PerfClassAdapter(ClassVisitor visitor, String theClass) {
        super(visitor);
        this.className = theClass;
    }
    public MethodVisitor visitMethod(int arg,
            String name,
            String descriptor,
            String signature,
            String[] exceptions) {
        MethodVisitor mv = super.visitMethod(arg,
                name,
                descriptor,
                signature,
                exceptions);
        MethodAdapter ma = new PerfMethodAdapter(mv, className, name);
        return ma;
    } 
}
PerfMethodAdapter.java
public class PerfMethodAdapter extends MethodAdapter {
    private String _className, _methodName;
   
    public PerfMethodAdapter(MethodVisitor visitor,
            String className,
            String methodName) {
        super(visitor);
        _className = className;
        _methodName = methodName;
    }

    public void visitCode() {
        this.visitLdcInsn(_className);
        this.visitLdcInsn(_methodName);
        this.visitMethodInsn(INVOKESTATIC,
                "
AOP_LOG",
                "start",
                "(Ljava/lang/String;Ljava/lang/String;)V");
        super.visitCode();
    }

    public void visitInsn(int inst) {
        switch (inst) {
        case Opcodes.ARETURN:
        case Opcodes.DRETURN:
        case Opcodes.FRETURN:
        case Opcodes.IRETURN:
        case Opcodes.LRETURN:
        case Opcodes.RETURN:
        case Opcodes.ATHROW:
            this.visitLdcInsn(_className);
            this.visitLdcInsn(_methodName);
            this.visitMethodInsn(INVOKESTATIC,
                    "AOP_LOG",
                    "end",
                    "(Ljava/lang/String;Ljava/lang/String;)V");
            break;
        default:
            break;
        }
        super.visitInsn(inst);
    }
}
AOP_LOG.java
public class
AOP_LOG{
   
    public static void start(String className, String methodName) {
        System.out.println(new StringBuilder(className)
                .append(‘\t‘)
                .append(methodName)
                .append("\tstart\t")
                .append(System.currentTimeMillis()));       
    }
    public static void end(String className, String methodName) {
        System.out.println(new StringBuilder(className)
                .append(‘\t‘)
                .append(methodName)
                .append("\tend\t")
                .append(System.currentTimeMillis()));       
    }
}
這些類結合起來,就實現(xiàn)了在方法的調用前,會調用
AOP_LOG類的start方法,在方法返回后,調用AOP_LOG類的end方法。

現(xiàn)在寫個測試類來測試一下。
TSMain.java
public class TSMain {

    public static void main(String[] args) {
        TSMain tsm = new TSMain();
        tsm.printClassName();
    }
   
    public void printClassName(){
        System.out.println("hello "+this.getClass().getName());
    }
}

運行命令如下:
=>java -javaagent:t.jar  classpath cpdir TSMain  #
cpdir 需要用到的類的路徑

輸出結果如下:
TSMain    main    start    1179374205562
TSMain    <init>    start    1179374205562
TSMain    <init>    end    1179374205562
TSMain    printClassName    start    1179374205562
hello TSMain
TSMain    printClassName    end    1179374205562
TSMain    main    end    1179374205562


到這里就完成整個aop的功能實現(xiàn)了。需要補充說明的是,這里修改類的字節(jié)碼是使用了asm的基礎類庫,所以需要導入這些asm的jar文件。
本站僅提供存儲服務,所有內容均由用戶發(fā)布,如發(fā)現(xiàn)有害或侵權內容,請點擊舉報。
打開APP,閱讀全文并永久保存 查看更多類似文章
猜你喜歡
類似文章
Java 將字符串動態(tài)生成字節(jié)碼的實現(xiàn)方法
spring用Aop實現(xiàn)計算器驗證(注解方式)
.Net平臺AOP技術研究
Android埋點方案的簡單實現(xiàn)-AOP之AspectJ
Java正則引發(fā)的思考 | 淘寶網(wǎng)綜合業(yè)務平臺團隊博客
[Q&A]Ruby中如何動態(tài)調用類及其方法
更多類似文章 >>
生活服務
分享 收藏 導長圖 關注 下載文章
綁定賬號成功
后續(xù)可登錄賬號暢享VIP特權!
如果VIP功能使用有故障,
可點擊這里聯(lián)系客服!

聯(lián)系客服