java字節(jié)碼框架ASM操作字節(jié)碼的方法淺析
之前我們已經(jīng)對ASM進行的詳細的介紹,需要的朋友們可以點擊這里:java字節(jié)碼框架ASM的深入學習
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);
對應的類型簽名就是
f (ILjava/lang/String;[I)J
再比如,java方法是
private void hi(double a, List<String> b);
那對應的類型簽名就是
hi (DLjava/util/List;)V
接下來可以利用ASM進行驗證上述兩個類型簽名是否正確:
public class Test { public static void main(String[] args) throws Exception { ClassPrinter printer = new ClassPrinter(); //讀取靜態(tài)內部類Bazhang ClassReader cr = new ClassReader("Test$Bazhang"); cr.accept(printer, 0); } //靜態(tà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); } } }
最后打印出來的內容:
java/lang/Object Test$Bazhang <init> ()V f (ILjava/lang/String;[I)J hi (DLjava/util/List;)V
驗證了之前的正確性,其中可以看到默認構造函數(shù)也打印出來了。
那么接下來干點有意思的事,我們往Bazhang類里新增和方法,就定為:
public void newFunc(String str){ }
這個時候就需要用到ClassWriter了,用于拼接字節(jié)碼,具體關于ClassReader、ClassVisitor、ClassWriter的文章可以查看這篇文章:ASM源碼學習之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文件對應的二進制流 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) { } }
結合之前整理的JVM指令集,使用ASM直接操作字節(jié)碼也是沒問題的,結尾附上ASM源碼下載地址:http://forge.ow2.org/projects/asm/
總結
以上就是這篇文章的全部內容了,希望本文的內容對大家的學習或者工作能帶來一定的幫助,如果有疑問大家可以留言交流。
- 學會Java字節(jié)碼指令,成為技術大佬
- Java之字節(jié)碼以及優(yōu)勢案例講解
- Java字節(jié)碼增強技術知識點詳解
- 詳解Java動態(tài)字節(jié)碼技術
- 詳解Java字節(jié)碼編程之非常好用的javassist
- 淺談javap命令拆解字節(jié)碼文件
- Java字節(jié)碼中jvm實例用法
- Javassist如何操作Java 字節(jié)碼
- Java中invokedynamic字節(jié)碼指令問題
- java獲取版本號及字節(jié)碼編譯版本方法示例
- java 獲取字節(jié)碼文件的幾種方法總結
- java 中如何獲取字節(jié)碼文件的相關內容
- java字節(jié)碼框架ASM的深入學習
- Java 將字符串動態(tài)生成字節(jié)碼的實現(xiàn)方法
- 通過java字節(jié)碼分析學習對象初始化順序
- Java字節(jié)碼的增強技術
相關文章
SpringBoot+Redis+Lua分布式限流的實現(xiàn)
本文主要介紹了SpringBoot+Redis+Lua分布式限流的實現(xiàn),文中通過示例代碼介紹的非常詳細,對大家的學習或者工作具有一定的參考學習價值,需要的朋友們下面隨著小編來一起學習學習吧2022-08-08解決問題:Failed to execute goal org.apache.m
這篇文章主要給大家介紹了關于解決問題:Failed to execute goal org.apache.maven.plugins:maven-resources-plugin:3.2.0:resources的相關資料,文中將解決的辦法介紹的非常詳細,需要的朋友可以參考下2023-03-03