Javaのバイトコードで遊ぶ (フィールドとメソッドの定義)
まずはフィールドの定義について。 ClassWriter.visitFieldを用います。 こんな感じです。
// public static int a = 100; cw.visitField(ACC_PUBLIC + ACC_STATIC, "a", "I", null, 100); // public static String; cw.visitField(ACC_PUBLIC + ACC_STATIC, "b", "Ljava/lang/String;", null, null); // public int c; cw.visitField(ACC_PUBLIC, "c", "I", null, null);
各フィールドの値はstaticブロックやコンストラクタで初期化するのが基本路線ですが、 staticなフィールドの場合はintやStringなどの型なら初期値を第4引数に指定可能です。
続いてメソッドの定義です。 ClassWriter.visitMethodで開始して、visitCode〜メソッドの内容〜visitMaxs〜visitEndという流れです。
mv = cw.visitMethod(ACC_PUBLIC + ACC_STATIC, "main", "([Ljava/lang/String;)V", null, null); mv.visitCode(); // // ここにメソッドの内容 // mv.visitMaxs(2, 1); mv.visitEnd();
staticブロックはメソッド名を<clinit>
、コンストラクタはメソッド名を<init>
にします。
// static {} mv = cw.visitMethod(ACC_STATIC, "<clinit>", "()V", null, null); // constructor mv = cw.visitMethod(ACC_PUBLIC, "<init>", "()V", null, null);
ジェネリック型や例外を使う場合は、visitMethodの第3引数、第4引数を使います。この記事では触れません。面倒なので。。
最後にサンプルコード。
package example; import org.objectweb.asm.ClassWriter; import org.objectweb.asm.MethodVisitor; import org.objectweb.asm.Opcodes; public class FieldAndMethod implements Opcodes { private static byte[] generate(String className) { ClassWriter cw = new ClassWriter(0); MethodVisitor mv; cw.visit(V1_8, ACC_PUBLIC + ACC_SUPER, className, null, "java/lang/Object", null); // fields // public static int a = 100; cw.visitField(ACC_PUBLIC + ACC_STATIC, "a", "I", null, 100); // public static String; cw.visitField(ACC_PUBLIC + ACC_STATIC, "b", "Ljava/lang/String;", null, null); // public int c; cw.visitField(ACC_PUBLIC, "c", "I", null, null); // static block mv = cw.visitMethod(ACC_STATIC, "<clinit>", "()V", null, null); mv.visitCode(); // -- System.out.println(a) mv.visitFieldInsn(GETSTATIC, "java/lang/System", "out", "Ljava/io/PrintStream;"); mv.visitFieldInsn(GETSTATIC, className, "a", "I"); mv.visitMethodInsn(INVOKEVIRTUAL, "java/io/PrintStream", "println", "(I)V", false); // -- b = "ABC" mv.visitLdcInsn("ABC"); mv.visitFieldInsn(PUTSTATIC, className, "b", "Ljava/lang/String;"); mv.visitInsn(RETURN); mv.visitMaxs(2, 0); mv.visitEnd(); // constructor mv = cw.visitMethod(ACC_PUBLIC, "<init>", "()V", null, null); mv.visitCode(); mv.visitVarInsn(ALOAD, 0); mv.visitMethodInsn(INVOKESPECIAL, "java/lang/Object", "<init>", "()V", false); // -- this.c = 200 mv.visitVarInsn(ALOAD, 0); mv.visitIntInsn(SIPUSH, 200); mv.visitFieldInsn(PUTFIELD, className, "c", "I"); mv.visitInsn(RETURN); mv.visitMaxs(3, 1); mv.visitEnd(); // main mv = cw.visitMethod(ACC_PUBLIC + ACC_STATIC, "main", "([Ljava/lang/String;)V", null, null); mv.visitCode(); // -- System.out.println(b) mv.visitFieldInsn(GETSTATIC, "java/lang/System", "out", "Ljava/io/PrintStream;"); mv.visitFieldInsn(GETSTATIC, className, "b", "Ljava/lang/String;"); mv.visitMethodInsn(INVOKEVIRTUAL, "java/io/PrintStream", "println", "(Ljava/lang/String;)V", false); mv.visitInsn(RETURN); mv.visitMaxs(2, 1); mv.visitEnd(); cw.visitEnd(); return cw.toByteArray(); } public static void main(String... args) throws Exception { String className = "FieldAndMethod"; byte[] bytes = generate(className); ExampleUtil.execMain(className, bytes); ExampleUtil.write(className, bytes); } }