java字節(jié)碼框架ASM操作字節(jié)碼的方法淺析
之前我們已經(jīng)對ASM進行的詳細(xì)的介紹,需要的朋友們可以點擊這里:java字節(jié)碼框架ASM的深入學(xué)習(xí)
JVM的類型簽名對照表
| Type Signature | Java Type |
|---|---|
| Z | boolean |
| B | byte |
| C | char |
| S | short |
| I | int |
| J | long |
| F | float |
| D | double |
| L | fully-qualified-class ;fully-qualified-class |
| [ type | type[] |
| ( arg-types ) ret-type | method type |
比如,java方法是
long f (int n, String s, int[] arr);
對應(yīng)的類型簽名就是
f (ILjava/lang/String;[I)J
再比如,java方法是
private void hi(double a, List<String> b);
那對應(yīng)的類型簽名就是
hi (DLjava/util/List;)V
接下來可以利用ASM進行驗證上述兩個類型簽名是否正確:
public class Test {
public static void main(String[] args) throws Exception {
ClassPrinter printer = new ClassPrinter();
//讀取靜態(tài)內(nèi)部類Bazhang
ClassReader cr = new ClassReader("Test$Bazhang");
cr.accept(printer, 0);
}
//靜態(tài)內(nèi)部類
static class Bazhang {
public Bazhang(int a) {
}
private long f (int n, String s, int[] arr){
return 0;
}
private void hi(double a, List<String> b){
}
}
static class ClassPrinter extends ClassVisitor {
public ClassPrinter() {
super(Opcodes.ASM5);
}
@Override
public void visit(int version, int access, String name, String signature, String superName, String[] interfaces) {
super.visit(version, access, name, signature, superName, interfaces);
//打印出父類name和本類name
System.out.println(superName + " " + name);
}
@Override
public MethodVisitor visitMethod(int access, String name, String desc, String signature, String[] exceptions) {
//打印出方法名和類型簽名
System.out.println(name + " " + desc);
return super.visitMethod(access, name, desc, signature, exceptions);
}
}
}
最后打印出來的內(nèi)容:
java/lang/Object Test$Bazhang <init> ()V f (ILjava/lang/String;[I)J hi (DLjava/util/List;)V
驗證了之前的正確性,其中可以看到默認(rèn)構(gòu)造函數(shù)也打印出來了。
那么接下來干點有意思的事,我們往Bazhang類里新增和方法,就定為:
public void newFunc(String str){
}
這個時候就需要用到ClassWriter了,用于拼接字節(jié)碼,具體關(guān)于ClassReader、ClassVisitor、ClassWriter的文章可以查看這篇文章:ASM源碼學(xué)習(xí)之ClassReader、ClassVisitor與ClassWriter詳解
public static void main(String[] args) throws Exception {
ClassReader cr = new ClassReader(Bazhang.class.getName());
ClassWriter cw = new ClassWriter(cr, ClassWriter.COMPUTE_MAXS);
cr.accept(cw, Opcodes.ASM5);
MethodVisitor mv = cw.visitMethod(ACC_PUBLIC, "newFunc", "(Ljava/lang/String;)V", null, null);
mv.visitInsn(Opcodes.RETURN);
mv.visitEnd();
// 獲取生成的class文件對應(yīng)的二進制流
byte[] code = cw.toByteArray();
//將二進制流寫到out/下
FileOutputStream fos = new FileOutputStream("out/Bazhang222.class");
fos.write(code);
fos.close();
}
這樣就會在out/文件夾下生成Bazhang222.class:
//
// Source code recreated from a .class file by IntelliJ IDEA
// (powered by Fernflower decompiler)
//
import java.util.List;
class Test$Bazhang {
Test$Bazhang() {
}
private long f(int n, String s, int[] arr) {
return 0L;
}
private void hi(double a, List<String> b) {
}
public void newFunc(String var1) {
}
}
結(jié)合之前整理的JVM指令集,使用ASM直接操作字節(jié)碼也是沒問題的,結(jié)尾附上ASM源碼下載地址:http://forge.ow2.org/projects/asm/
總結(jié)
以上就是這篇文章的全部內(nèi)容了,希望本文的內(nèi)容對大家的學(xué)習(xí)或者工作能帶來一定的幫助,如果有疑問大家可以留言交流。
- 學(xué)會Java字節(jié)碼指令,成為技術(shù)大佬
- Java之字節(jié)碼以及優(yōu)勢案例講解
- Java字節(jié)碼增強技術(shù)知識點詳解
- 詳解Java動態(tài)字節(jié)碼技術(shù)
- 詳解Java字節(jié)碼編程之非常好用的javassist
- 淺談javap命令拆解字節(jié)碼文件
- Java字節(jié)碼中jvm實例用法
- Javassist如何操作Java 字節(jié)碼
- Java中invokedynamic字節(jié)碼指令問題
- java獲取版本號及字節(jié)碼編譯版本方法示例
- java 獲取字節(jié)碼文件的幾種方法總結(jié)
- java 中如何獲取字節(jié)碼文件的相關(guān)內(nèi)容
- java字節(jié)碼框架ASM的深入學(xué)習(xí)
- Java 將字符串動態(tài)生成字節(jié)碼的實現(xiàn)方法
- 通過java字節(jié)碼分析學(xué)習(xí)對象初始化順序
- Java字節(jié)碼的增強技術(shù)
相關(guān)文章
JAVA中使用FileWriter寫數(shù)據(jù)到文本文件步驟詳解
這篇文章主要介紹了JAVA中使用FileWriter寫數(shù)據(jù)到文本文件步驟詳解,FileWriter類提供了多種寫入字符的方法,包括寫入單個字符、寫入字符數(shù)組和寫入字符串等,它還提供了一些其他的方法,如刷新緩沖區(qū)、關(guān)閉文件等,需要的朋友可以參考下2023-10-10
SpringBoot中如何解決讀取properties文件讀取問題
這篇文章主要介紹了SpringBoot中如何解決讀取properties文件讀取問題,具有很好的參考價值,希望對大家有所幫助。如有錯誤或未考慮完全的地方,望不吝賜教2022-07-07

