chi*_*han 2 java java-bytecode-asm
我使用Java ASM编写一个简单的程序来生成通过编译以下类生成的字节代码.
public class Main {
public static void main(String[] args) {
System.out.println("Test");
}
}
Run Code Online (Sandbox Code Playgroud)
我编写的用于生成此类的字节码的代码如下所示.
public class CodeGenerator {
public void generateClass()
{
ClassWriter cw=new ClassWriter(Opcodes.NULL);
FieldVisitor fv;
MethodVisitor mv;
cw.visit(Opcodes.V1_1, Opcodes.ACC_PUBLIC, "Main", null, "java/lang/Object", null);
mv=cw.visitMethod(Opcodes.ACC_PUBLIC, "<init>", "()V", null, null);
mv.visitVarInsn(Opcodes.AALOAD, 0);
mv.visitMethodInsn(Opcodes.INVOKESPECIAL, "java/lang/Object", "<init>", "()V");
mv.visitInsn(Opcodes.RETURN);
mv.visitMaxs(1, 1);
mv.visitEnd();
mv=cw.visitMethod(Opcodes.ACC_PUBLIC+ Opcodes.ACC_STATIC, "Main", "([Ljava/lang/String;)V", null, null);
mv.visitFieldInsn(Opcodes.GETSTATIC, "java/lang/System", "out", "Ljava/io/PrintStream");
mv.visitLdcInsn("Test");
mv.visitMethodInsn(Opcodes.INVOKEVIRTUAL, "java/io/PrintStream", "println", "(Ljava/lang/String;)V");
mv.visitInsn(Opcodes.RETURN);
mv.visitMaxs(2, 1);
mv.visitEnd();
cw.visitEnd();
this.WriteClass(cw);
}
public void WriteClass(ClassWriter cw){
FileOutputStream fos;
try{
fos = new FileOutputStream("E:\\Acadamic\\Final year project\\ASM\\Main.class");
fos.write(cw.toByteArray());
fos.close();
}
catch (IOException ex){
Logger.getLogger(CodeGenerator.class.getName()).log(Level.SEVERE, null, ex);
}
}
}
Run Code Online (Sandbox Code Playgroud)
这成功编译,当我尝试运行它时,它会给出以下错误.
Exception in thread "main" java.lang.NoClassDefFoundError: main/class
Caused by: java.lang.ClassNotFoundException: main.class
at java.net.URLClassLoader$1.run(Unknown Source)
at java.security.AccessController.doPrivileged(Native Method)
at java.net.URLClassLoader.findClass(Unknown Source)
at java.lang.ClassLoader.loadClass(Unknown Source)
at sun.misc.Launcher$AppClassLoader.loadClass(Unknown Source)
at java.lang.ClassLoader.loadClass(Unknown Source)
Could not find the main class: main.class. Program will exit.
Run Code Online (Sandbox Code Playgroud)
任何人都可以帮助我避免这个错误.如果有人能够解释错误的原因以及如何避免错误,请善待我.
谢谢.
跟进:
当我像java Main一样运行它时会出现以下错误.
Exception in thread "main" java.lang.ClassFormatError: Field "out" in class Main
has illegal signature "Ljava/io/PrintStream"
at java.lang.ClassLoader.defineClass1(Native Method)
at java.lang.ClassLoader.defineClassCond(Unknown Source)
at java.lang.ClassLoader.defineClass(Unknown Source)
at java.security.SecureClassLoader.defineClass(Unknown Source)
at java.net.URLClassLoader.defineClass(Unknown Source)
at java.net.URLClassLoader.access$000(Unknown Source)
at java.net.URLClassLoader$1.run(Unknown Source)
at java.security.AccessController.doPrivileged(Native Method)
at java.net.URLClassLoader.findClass(Unknown Source)
at java.lang.ClassLoader.loadClass(Unknown Source)
at sun.misc.Launcher$AppClassLoader.loadClass(Unknown Source)
at java.lang.ClassLoader.loadClass(Unknown Source)
Could not find the main class: Main. Program will exit.
Run Code Online (Sandbox Code Playgroud)
Ada*_*ter 10
我不认为您的错误是由ASM引起的.看起来你只是java错误地调用了.你应该这样称呼它:
java Main
我猜你是这样叫的:
java Main.class
您的代码有三个问题.
该ClassWriter构造函数接受的选项的位掩码.你通过了它Opcodes.NULL.虽然这似乎是正确的,但你实际上想要传递它0(Opcodes.NULL意味着不同的东西).
ClassWriter cw=new ClassWriter(0);
Run Code Online (Sandbox Code Playgroud)
您正在调用AALOAD构造函数而不是ALOAD.该AALOAD指令加载数组元素,然后ALOAD加载局部变量.
mv.visitVarInsn(Opcodes.ALOAD, 0);
Run Code Online (Sandbox Code Playgroud)
在你的Main方法中,你忘记了分号后Ljava/io/PrintStream:
mv.visitFieldInsn(
Opcodes.GETSTATIC,
"java/lang/System",
"out",
"Ljava/io/PrintStream;" // Notice the semicolon
);
Run Code Online (Sandbox Code Playgroud)
你可以利用ASM的Type课程来节省一些痛苦:
mv.visitFieldInsn(
Opcodes.GETSTATIC,
"java/lang/System",
"out",
Type.getObjectType("java/io/PrintStream").getDescriptor()
);
Run Code Online (Sandbox Code Playgroud)