詳解Java中的字節(jié)碼增強(qiáng)技術(shù)
1.字節(jié)碼增強(qiáng)技術(shù)
字節(jié)碼增強(qiáng)技術(shù)就是一類(lèi)對(duì)現(xiàn)有字節(jié)碼進(jìn)行修改或者動(dòng)態(tài)生成全新字節(jié)碼文件的技術(shù)。
2.常見(jiàn)技術(shù)
| 技術(shù)分類(lèi) | 類(lèi)型 |
|---|---|
| 靜態(tài)增強(qiáng) | AspectJ |
| 動(dòng)態(tài)增強(qiáng) | ASM、Javassist、Cglib、Java Proxy |

3.ASM
<dependency>
<groupId>org.ow2.asm</groupId>
<artifactId>asm</artifactId>
<version>9.4</version>
</dependency>
ASM Core API可以類(lèi)比解析XML文件中的SAX方式,不需要把這個(gè)類(lèi)的整個(gè)結(jié)構(gòu)讀取進(jìn)來(lái),就可以用流式的方法來(lái)處理字節(jié)碼文件。好處是非常節(jié)約內(nèi)存,但是編程難度較大。然而出于性能考慮,一般情況下編程都使用Core API。在Core API中有以下幾個(gè)關(guān)鍵類(lèi):
| 技術(shù)分類(lèi) | 類(lèi)型 |
|---|---|
| ClassReader | 用于讀取已經(jīng)編譯好的.class文件。 |
| ClassWriter | 用于重新構(gòu)建編譯后的類(lèi),如修改類(lèi)名、屬性以及方法,也可以生成新的類(lèi)的字節(jié)碼文件。 |
| Visitor類(lèi) | 如上所述,CoreAPI根據(jù)字節(jié)碼從上到下依次處理,對(duì)于字節(jié)碼文件中不同的區(qū)域有不同的Visitor,比如用于訪(fǎng)問(wèn)方法的MethodVisitor、用于訪(fǎng)問(wèn)類(lèi)變量的FieldVisitor、用于訪(fǎng)問(wèn)注解的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ù)的文章就介紹到這了,更多相關(guān)Java字節(jié)碼增強(qiáng)內(nèi)容請(qǐng)搜索腳本之家以前的文章或繼續(xù)瀏覽下面的相關(guān)文章希望大家以后多多支持腳本之家!
相關(guān)文章
如何利用SpringAOP的返回通知處理數(shù)據(jù)加密返回
這篇文章主要介紹了如何利用SpringAOP的返回通知處理數(shù)據(jù)加密返回,本文通過(guò)實(shí)例代碼給大家介紹的非常詳細(xì),感興趣的朋友一起看看吧2024-12-12
Java Swing JCheckBox復(fù)選框的實(shí)現(xiàn)方法
這篇文章主要介紹了Java Swing JCheckBox復(fù)選框的實(shí)現(xiàn)方法,文中通過(guò)示例代碼介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友們下面隨著小編來(lái)一起學(xué)習(xí)學(xué)習(xí)吧2019-12-12
RestTemplate實(shí)現(xiàn)發(fā)送帶headers的GET請(qǐng)求
這篇文章主要介紹了RestTemplate實(shí)現(xiàn)發(fā)送帶headers的GET請(qǐng)求,具有很好的參考價(jià)值,希望對(duì)大家有所幫助。如有錯(cuò)誤或未考慮完全的地方,望不吝賜教2021-10-10
利用MultipartFile實(shí)現(xiàn)文件上傳功能
這篇文章主要為大家詳細(xì)介紹了利用MultipartFile實(shí)現(xiàn)文件上傳功能,具有一定的參考價(jià)值,感興趣的小伙伴們可以參考一下2017-11-11
使用SpringBoot簡(jiǎn)單實(shí)現(xiàn)無(wú)感知的刷新 Token功能
實(shí)現(xiàn)無(wú)感知的刷新 Token 是一種提升用戶(hù)體驗(yàn)的常用技術(shù),可以在用戶(hù)使用應(yīng)用時(shí)自動(dòng)更新 Token,無(wú)需用戶(hù)手動(dòng)干預(yù),這種技術(shù)在需要長(zhǎng)時(shí)間保持用戶(hù)登錄狀態(tài)的應(yīng)用中非常有用,以下是使用Spring Boot實(shí)現(xiàn)無(wú)感知刷新Token的一個(gè)場(chǎng)景案例和相應(yīng)的示例代碼2024-09-09
關(guān)于任務(wù)調(diào)度框架quartz使用(異常處理,解決恢復(fù)后多次調(diào)度處理)
這篇文章主要介紹了關(guān)于任務(wù)調(diào)度框架quartz使用(異常處理,解決恢復(fù)后多次調(diào)度處理),具有很好的參考價(jià)值,希望對(duì)大家有所幫助,如有錯(cuò)誤或未考慮完全的地方,望不吝賜教2023-12-12
Mybatis-plus操作json字段實(shí)戰(zhàn)教程
這篇文章主要介紹了Mybatis-plus操作json字段實(shí)戰(zhàn)教程,本文結(jié)合實(shí)例代碼給大家介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或工作具有一定的參考借鑒價(jià)值,需要的朋友可以參考下2023-02-02
使用Java實(shí)現(xiàn)MapReduce詞頻統(tǒng)計(jì)示例代碼
這篇文章主要介紹了使用Java實(shí)現(xiàn)MapReduce詞頻統(tǒng)計(jì)的相關(guān)資料,通過(guò)詞頻統(tǒng)計(jì)示例來(lái)展示MapReduce的運(yùn)行機(jī)制,涵蓋了Mapper和Reducer的實(shí)現(xiàn),并說(shuō)明了如何配置和執(zhí)行MapReduce作業(yè),需要的朋友可以參考下2024-11-11

