之前我們已經對ASM進行的詳細的介紹,需要的朋友們可以點擊這里:java字節碼框架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(); //讀取靜態內部類Bazhang ClassReader cr = new ClassReader("Test$Bazhang"); cr.accept(printer, 0); } //靜態內部類 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> ()Vf (ILjava/lang/String;[I)Jhi (DLjava/util/List;)V
驗證了之前的正確性,其中可以看到默認構造函數也打印出來了。
那么接下來干點有意思的事,我們往Bazhang類里新增和方法,就定為:
public void newFunc(String str){ }這個時候就需要用到ClassWriter了,用于拼接字節碼,具體關于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直接操作字節碼也是沒問題的,結尾附上ASM源碼下載地址:http://forge.ow2.org/projects/asm/
總結
以上就是這篇文章的全部內容了,希望本文的內容對大家的學習或者工作能帶來一定的幫助,如果有疑問大家可以留言交流。
新聞熱點
疑難解答