Maison >Java >javaDidacticiel >Une brève analyse de la méthode d'exploitation du bytecode dans le framework java bytecode ASM
Nous avons déjà présenté ASM en détail. Les amis qui en ont besoin peuvent cliquer ici : Étude approfondie du framework de bytecode Java ASM
Tableau de comparaison des signatures de type JVM
Par exemple, la méthode Java est
long f (int n, String s, int[] arr);
et la signature de type correspondante est
f (ILjava/lang/String;[I)J
Pour un autre exemple, si la méthode Java est
private void hi(double a, List<String> b);
, alors la signature de type correspondante est
hi (DLjava/util/List;)V
Ensuite, vous pouvez utiliser ASM pour vérification Les deux types de signatures ci-dessus sont-ils corrects :
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); } } }
Le contenu final imprimé :
java/lang/Object Test$Bazhang <init> ()V f (ILjava/lang/String;[I)J hi (DLjava/util/List;)V
L'exactitude précédente a été vérifiée et vous pouvez voir que le constructeur par défaut est également imprimé.
Ensuite, faisons quelque chose d'intéressant. Ajoutons une nouvelle méthode à la classe Bazhang, qui sera :
public void newFunc(String str){ }
Vous devez l'utiliser à ce moment-là. ClassWriter est utilisé pour épisser le bytecode. Pour des articles spécifiques sur ClassReader, ClassVisitor et ClassWriter, vous pouvez consulter cet article : Explication détaillée de ClassReader, ClassVisitor et ClassWriter pour l'apprentissage du code source ASM
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(); }
Cela générera Bazhang222.class dans le dossier out/ :
// // 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) { } }
Combiné avec le jeu d'instructions JVM précédemment organisé , utilisez Il n'y a aucun problème pour ASM à exploiter directement le bytecode. À la fin, l'adresse de téléchargement du code source ASM est jointe : http://forge.ow2.org/projects/asm/
Résumé
.C'est tout. L'intégralité du contenu de l'article est inclus. J'espère que le contenu de cet article pourra être utile aux études ou au travail de chacun. Si vous avez des questions, vous pouvez laisser un message pour communiquer.
Pour plus d'articles sur les méthodes du framework de bytecode Java ASM pour faire fonctionner le bytecode, veuillez faire attention au site Web PHP chinois !