溫馨提示×

您好,登錄后才能下訂單哦!

密碼登錄×
登錄注冊(cè)×
其他方式登錄
點(diǎn)擊 登錄注冊(cè) 即表示同意《億速云用戶服務(wù)條款》

Java中的字節(jié)碼增強(qiáng)技術(shù)是什么

發(fā)布時(shí)間:2022-10-12 10:21:33 來源:億速云 閱讀:124 作者:iii 欄目:開發(fā)技術(shù)

這篇“Java中的字節(jié)碼增強(qiáng)技術(shù)是什么”文章的知識(shí)點(diǎn)大部分人都不太理解,所以小編給大家總結(jié)了以下內(nèi)容,內(nèi)容詳細(xì),步驟清晰,具有一定的借鑒價(jià)值,希望大家閱讀完這篇文章能有所收獲,下面我們一起來看看這篇“Java中的字節(jié)碼增強(qiáng)技術(shù)是什么”文章吧。

1.字節(jié)碼增強(qiáng)技術(shù)

字節(jié)碼增強(qiáng)技術(shù)就是一類對(duì)現(xiàn)有字節(jié)碼進(jìn)行修改或者動(dòng)態(tài)生成全新字節(jié)碼文件的技術(shù)。

2.常見技術(shù)

技術(shù)分類類型
靜態(tài)增強(qiáng)AspectJ
動(dòng)態(tài)增強(qiáng)ASM、Javassist、Cglib、Java Proxy

Java中的字節(jié)碼增強(qiáng)技術(shù)是什么

3.ASM

<dependency>
    <groupId>org.ow2.asm</groupId>
    <artifactId>asm</artifactId>
    <version>9.4</version>
</dependency>

ASM Core API可以類比解析XML文件中的SAX方式,不需要把這個(gè)類的整個(gè)結(jié)構(gòu)讀取進(jìn)來,就可以用流式的方法來處理字節(jié)碼文件。好處是非常節(jié)約內(nèi)存,但是編程難度較大。然而出于性能考慮,一般情況下編程都使用Core API。在Core API中有以下幾個(gè)關(guān)鍵類:

技術(shù)分類類型
ClassReader用于讀取已經(jīng)編譯好的.class文件。
ClassWriter用于重新構(gòu)建編譯后的類,如修改類名、屬性以及方法,也可以生成新的類的字節(jié)碼文件。
Visitor類如上所述,CoreAPI根據(jù)字節(jié)碼從上到下依次處理,對(duì)于字節(jié)碼文件中不同的區(qū)域有不同的Visitor,比如用于訪問方法的MethodVisitor、用于訪問類變量的FieldVisitor、用于訪問注解的AnnotationVisitor等。為了實(shí)現(xiàn)AOP,重點(diǎn)要使用的是MethodVisitor。

3.1 測(cè)試 Main

package com.xu.test;


/**
 * @author Administrator
 */
public class Main {

    public void print() {
        System.out.println("ASM");
    }

}

3.2 測(cè)試 CustomerClassVisitor

package com.xu.test;

import org.apache.commons.lang3.StringUtils;
import org.objectweb.asm.ClassVisitor;
import org.objectweb.asm.MethodVisitor;
import org.objectweb.asm.Opcodes;

/**
 * ASM 字節(jié)碼增強(qiáng)技術(shù)
 *
 * @author Administrator
 */
public class CustomerClassVisitor extends ClassVisitor implements Opcodes {

    public CustomerClassVisitor(ClassVisitor api) {
        super(ASM9, api);
    }

    @Override
    public void visit(int version, int access, String name, String signature, String superName, String[] interfaces) {
        cv.visit(version, access, name, signature, superName, interfaces);
    }

    @Override
    public MethodVisitor visitMethod(int access, String name, String descriptor, String signature, String[] exceptions) {
        MethodVisitor mv = cv.visitMethod(access, name, descriptor, signature, exceptions);
        if (StringUtils.equals("print", name) && mv != null) {
            mv = new CustomerMethodVisitor(mv);
        }
        return mv;
    }

    class CustomerMethodVisitor extends MethodVisitor implements Opcodes {
        public CustomerMethodVisitor(MethodVisitor api) {
            super(ASM9, api);
        }

        @Override
        public void visitCode() {
            super.visitCode();
            mv.visitFieldInsn(GETSTATIC, "java/lang/System", "out", "Ljava/io/PrintStream;");
            mv.visitLdcInsn("start");
            mv.visitMethodInsn(INVOKEVIRTUAL, "java/io/PrintStream", "println", "(Ljava/lang/String;)V", false);
        }

        @Override
        public void visitInsn(int opcode) {
            if ((opcode >= Opcodes.IRETURN && opcode <= Opcodes.RETURN) || opcode == Opcodes.ATHROW) {
                mv.visitFieldInsn(GETSTATIC, "java/lang/System", "out", "Ljava/io/PrintStream;");
                mv.visitLdcInsn("end");
                mv.visitMethodInsn(INVOKEVIRTUAL, "java/io/PrintStream", "println", "(Ljava/lang/String;)V", false);
            }
            mv.visitInsn(opcode);
        }
    }

}

3.3 測(cè)試 Test

package com.xu.test;

import java.io.File;
import java.io.FileOutputStream;

import org.objectweb.asm.ClassReader;
import org.objectweb.asm.ClassVisitor;
import org.objectweb.asm.ClassWriter;

/**
 * @author Administrator
 */
public class Test {

    public static void main(String[] args) throws Exception {
        ClassReader reader = new ClassReader("com/xu/test/Main");
        ClassWriter writer = new ClassWriter(ClassWriter.COMPUTE_MAXS);
        // 處理
        ClassVisitor visitor = new CustomerClassVisitor(writer);
        reader.accept(visitor, ClassReader.SKIP_DEBUG);
        // 輸出
        File file = new File("target\\classes\\com\\xu\\test\\Main.class");
        FileOutputStream stream = new FileOutputStream(file);
        stream.write(writer.toByteArray());
        stream.close();
        // 測(cè)試
        Class<?> cls = Class.forName("com.xu.test.Main");
        Main main = (Main) cls.getDeclaredConstructor().newInstance();
        main.print();
    }


}

以上就是關(guān)于“Java中的字節(jié)碼增強(qiáng)技術(shù)是什么”這篇文章的內(nèi)容,相信大家都有了一定的了解,希望小編分享的內(nèi)容對(duì)大家有幫助,若想了解更多相關(guān)的知識(shí)內(nèi)容,請(qǐng)關(guān)注億速云行業(yè)資訊頻道。

向AI問一下細(xì)節(jié)

免責(zé)聲明:本站發(fā)布的內(nèi)容(圖片、視頻和文字)以原創(chuàng)、轉(zhuǎn)載和分享為主,文章觀點(diǎn)不代表本網(wǎng)站立場(chǎng),如果涉及侵權(quán)請(qǐng)聯(lián)系站長(zhǎng)郵箱:is@yisu.com進(jìn)行舉報(bào),并提供相關(guān)證據(jù),一經(jīng)查實(shí),將立刻刪除涉嫌侵權(quán)內(nèi)容。

AI