在这一节中将会介绍java的基本操作在asmsupport的实现。在介绍操作之前,我们首先确定我们有哪些基本的操作,可以参考下表
除了上述的操作,还有一下操作
我们先介绍三个基础的操作1。获取常量值,2.获取AClass对象,3.创建变量,然后再按照上述顺序自上而下介绍。我们将会使用asmsupport先创建一个类OperatorTest,同时在这个类中添加一个main方法。在这里我们会提前使用到如何创建一个class或者method的知识,但是这并不是本节的重点,创建class/enum/interface等将在后面做详细介绍。生产的类和方法大致如下:
public class OperatorTest { public static void main(String[] args) { //我们将用asmsupport在这里位置生产我们期望的内容 } }
DummyClass dc = new DummyClass("asmsupport.OperatorTest").public_(); dc.newMethod("main").public_().static_().argTypes(Stirng[].class)._argNames("args") .body(new MethodBody(){ @Override public void body(LocVar... args) { //在这里调用相应的操作生产字节码。 //TODO } }); Class cls = dc.builder();
在上面的代码中我们就通过asmsupport创建出一个class的简单框架。我们也将在TODO位置生产我们预期的内容,也就是说下面我们介绍的操作的所有代码都将是在这个位置调用的,而在下面介绍操作的时候,将省去类创建的过程。
这里介绍asmsupport如何获得常量值,在asmsupport可以直接使用的值包括基本类型的值(boolean, byte, char, short, int, long, float, double), String类型,Class类型,以及null(null值允许是任何非基本类型).在asmsupport这些操作主要通过两种方式获得。
直接通过Value类的静态方法value获取非null值,获取null值通过方法getNullValue。代码如下:
IValue one = Value.value(1); //获取一个int类型的值1 IValue oneDotOne = Value.value(2.2D); //获取一个double类型的值2.2 IValue str = Value.value("asmsupport"); //获取一个String类型的"asmsupport"值 IValue nullVal = Value.getNullValue(String.class); //获取String类型的
如果你的代码已经在一个程序块中了,正如我们的实例,我们的代码在TODO的位置,那么我们就可以使用另一种方式,通过ValueAction接口来获取IValue对象。
IValue one =val(1); //获取一个int类型的值1 IValue oneDotOne = val(2.2D); //获取一个double类型的值2.2 IValue str = val("asmsupport"); //获取一个String类型的"asmsupport"值 IValue nullVal = null_(String.class); //获取String类型的
上面的Value.val方法和ValueAction.val方法均为重载方法,具体希望获取什么类型的值取决于你传入的参数。
上面介绍的是如何创建常量值,这里我们介绍如何获取一个数组值。通常在java中创建数组的时候一般有两种方式如下
new int[1][2][]; new String[]{"Hello", "ASMSupport"}
上面两种方式一个只创建了数组空间,而第二个则是在创建数组的时候就赋予了值。那么接下来就是使用asmsupport创建上面两个值,我们使用ArrayAction接口中的makeArray和newarray两个方法来实现。
makeArray(int[][][].class, val(1), val(2)); newarray(String[].class, new Parameterized[]{val("Hello"), val("ASMSupport")});
该接口下还有很多重载方法。
通常我们在使用asmsupport的时候,很多方法都直接使用Class对象即可,但是还有部分操作是要通过AClass完成的,比如获取静态field,和常量值一样,这里同意有两种方式获取AClass。
通过AClassFactory类,这个类中包含了多个静态方法。
1. AClass arrayList = AClassFactory.deftype(ArrayList.class); 2. AClass stringDoubleDimArray = AClassFactory.deftype(String[][].class); 3. AClass stringDoubleDimArray = AClassFactory.defArrayType(String[][].class); 4. AClass stringDoubleDimArray = AClassFactory.defArrayType(String.class, 2); 5. AClass stringDoubleDimArray = AClassFactory.defArrayType(AClassFactory.deftype(String.class), 2);
如果你的代码在一个asmsupport的程序块中,比如上面代码中的TODO位置,我们可以通过AClassDefAction接口来获取
1. AClass arrayList = deftype(ArrayList.class); 2. AClass stringDoubleDimArray = deftype(String[][].class); 3. AClass stringDoubleDimArray = defArrayType(String[][].class); 4. AClass stringDoubleDimArray = defArrayType(String.class, 2); 5. AClass stringDoubleDimArray = defArrayType(AClassFactory.deftype(String.class), 2);
上面的例子中,第一行创建的是一个ArrayList的AClass,2~5都是创建一个ArrayClass,都是String类型的一个二维数组。
通过VariableAction接口的var方法即可创建局部变量。
//创建String类型变量name,并且赋值为"asmsupport" //String name = "asmsupport"; 1. LocVar name = var("name", String.class, val("asmsupport")); //创建一个String数组 //String[] names = new String[]{"Hello", "ASMSupport"}; 2. LocVar names = arrayvar("names", String[].class, newArray(String[].class, new Parameterized[]{val("Hello"), val("ASMSupport")}););
这个两个操作是通过CrementAction接口完成,该接口是完成自增减操作。
1. LocVar i = var("i", int.class, val(1)); 2. postinc(i); //i++ 3. postdec(i); //i--
这里的操作都是一元操作,其中++expr –expr依然采用CrementAction接口的方法完成。+expr不需要任何操作,-expr是取负数采用ActionSet接口中的neg方法。~则是取反操作,使用BitwiseAction的reverse方法。
1. LocVar i = var("i", int.class, val(8)); 2. perinc(i); // ++i; 3. perdec(i); // --i; 4. neg(i); // -i; 5. reverse(i); // ~i
这里实现了乘法除法取模操作,其方法是在ArithmeticAction接口中
1. LocVar i = var("i", int.class, val(8)); 2. LocVar j = var("i", int.class, val(2)); 3. mul(i, j); // i * j 4. div(i, j); // i / j 5. mod(i, j); // i % j
加法减法操作
1. LocVar i = var("i", int.class, val(8)); 2. LocVar j = var("i", int.class, val(2)); 3. add(i, j); // i + j 4. sub(i, j); // i - j
这里的操作实现了位移操作分别是左移,右移,无符号右移,通过BitwiseAction接口的方法完成
1. LocVar i = var("i", int.class, val(8)); 2. shl(i, val(2)); // i << 2 3. shr(i, val(2)); // i >> 2 4. ushr(i, val(2)); // i >>> 2
这组的操作关系操作,其中这一组(<,>,<=,>=)是使用RelationalAction接口,instanceof使用的是ActionSet接口的instanceof_方法。
1. LocVar i = var("i", int.class, val(8)); 2. LocVar j = var("i", int.class, val(2)); 3. lt(i, j); // i < j 4. gt(i, j); // i > j 5. le(i, j); // i <= j 6. ge(i, j); // i >= j 7. LocVar list = var("list", List.class, new(ArrayList.class)) 8. instanceof_(list, ArrayList.class);(
1. LocVar i = var("i", int.class, val(8)); 2. LocVar j = var("i", int.class, val(2)); 3. eq(i, j); // i == j 4. ne(i, j); // i != j)
这操同样能够比较两个对象比如obj1 == obj2或obj1 != obj2. 但是这个操作并不是会调用Object的equals的方法。
这里的的与操作存在两种含义,一个是对数字类型的进行与运算,比如1&2,还有就是对于boolean类型的的逻辑与,但底层是同样的字节码实现。这样的两种操作很容易操作失误护着产生歧义,后期版本将会修复。
1. LocVar i = var("i", int.class, val(8)); 2. LocVar j = var("i", int.class, val(2)); 3. LocVar m = var("i", boolean.class, val(true)); 4. LocVar n = var("i", boolean.class, val(false)); 5. band(i, j); // i & j 6. logicAnd(m, n); // m & n
这个操作完成的是三元操作,使用的是ActionSet接口的ternary方法
1. ternary(gt(i, j), sub(i, j), add(i, j));// i > j ? i - j : i + j
这里的赋值操作非常多,但是asmsupport仅仅实现了单一的赋值操作,其余的都是先执行运算再赋值大同小异。使用的是VariableAction接口中的assign.
1. LocVar i = var("i", int.class, val(1)); 2. assign(i, val(10));
之前已经介绍如何创建一个数组,这里将介绍如何对数组操作,比如获取数组元素,获取数组长度,保存数组元素,这些操作均来自与接口ArrayAction接口
1. LocVar array = arrayvar("i", int[][].class, makeArray(int[][].class, val(2), val(2)));//int[][] i = new int[2][2]; 2. arrayStore(array, val(100), val(1), val(1)); //array[1][1] = 100; 3. arrayLoad(array, val(1), val(1)); //array[1][1] 4. arrayLength(array, val[1]); //array[1].length
在中我们可以使用return返回,如果该方法有返回值,则需要在return后添加返回值。在asmsupport中使用ActionSet接口中的return_()方法和return_(returnObj).
1. return_(); //return; 2. return_(add(val(1), val(2))); //return 1+2;
这里面需要注意的是,在java代码中一个没有返回值的方法,如果没有显式的执行return,那么java编译器会自动的在方法最后追加return操作。但是在asmsupport中并没这样的判断,所以需要显式的执行return方法。这一点要注意,后期版本将会支持。
有两种field获取,一个静态field,另一个是非静态field。和java一样静态方法通过AClass对象的field方法获取,非静态方法通过IVariable对象的field方法获取。但是如果通过 IVariable对象获取静态field同样也允许的。比如有下面这个类
public class A { public String field1 = "Hello"; public static String field2 = "ASMSupport"; }
下面是asmsupport获取field的方式
1. LocVar a = var("a", new_(A.class)); //A a = new A(); 2. a.field("field1"); // a.field1 3. a.field("field2"); // A.field2 4. defType(A.class).field("field2"); //A.field2
第2行是获取非静态field,第3,4行分别是通过变量和Class获取静态field。
总共有三种类型的方法调用:1.调用构造方法,2.调用对象的方法,3.调用静态方法。这些操作是在MethodInvokeAction接口中
1. LocVar list = var("list", List.class, new_(ArrayList.class)); //创建一个ArrayList对象,new ArrayList(); 2. call(list, "add", val(1)); // list.add(1); 3. call(Integer.class, "parseInt", val("100")); // Integer.parseInt("100")
第1行调用构造方法,第2行调用非静态方法,第3行调用静态方法。