目录
我们都知道:
b++ : ++后置的话,先使用,在计算
--c: ++前置 的话.先计算,在使用
字节码分析:
那本质原因是啥呢,今天就来分析一下:
0 iconst_0 --把int类型的 0 压入操作数栈, 即b
1 istore_1 -- 弹出栈顶元素 0 ,存入位置为1的局部变量中
2 iconst_0 --把int类型的 0 压入操作数栈, 即c
3 istore_2 -- 弹出栈顶元素 0 ,存入位置为1的局部变量中
--重点来了啊
//这里是后置的字节码
4 iload_1 -- 从位置1的局部变量中取出元素:0 压入操作数栈中
5 iinc 1 by 1 -- 把位置1的局部变量 进行 +1
8 istore_2 -- 弹出栈顶元素,存入位置为 2的局部变量中
//这里是前置的字节码
9 iinc 2 by -1 -- 把位置2的局部变量 进行 -1
12 iload_2 -- 从位置2的局部变量中取出元素:-1 压入操作数栈中
13 istore_1 -- 弹出栈顶元素,存入位置为 1的局部变量中
所以,总结:
如果是后置的字节码的话:
1.先把局部变量的元素压入栈中 iload_1
2.在把局部变量进行 +1, iinc 1 by 1
3.最后把栈顶元素弹出存入局部变量 istore_2
如果是前置的字节码的话 :
1.先把局部变量的元素进行-1, iinc 2 by -1
2.在把局部变量的元素压入操作数栈 iload_2
3.再把栈顶元素弹出存入局部变量中 istore_1
看一下下面的图,已经能够理解的比较深刻:
总结:
前置的字节码,是先把局部变量的x位置的元素 进行 +1,再把 元素压入栈,所以这里取到的是+1操作后的值,再把栈顶值弹出保存到局部变量表中
后置的字节码,是先把局部变量的x位置元素取出,压入栈里面,所以这里拿的值是原来的值,
再把元素的值+1, 再把栈顶值弹出保存到局部变量表中,这里栈顶就是原值
有不正确的地方欢迎指出!!
(有没有疑惑,局部变量表0的位置存放的是什么呢?,下面就来解惑一下)
那局部变量表位置为0 存放的是啥呢??
我们来反编译看一下下面的代码:
public class Test2 {
private int d = 0;
//非静态方法
public void calc(){
this.d = 1;
int i =0;
}
//静态方法
public static void staticCalc(){
int i =0;
}
//main方法
public static void main(String[] args) {
int c= 0;
}
//带参数 静态方法
public static void staticCalcArgs(int x){
int i =0;
}
}
先使用
javac -g:vars Demo3_11_3.java 这样的话就能查看局部变量表的信息了
再
javap -v Demo3_11_3.class
重新查看字节码信息
看下图:
main方法
带参数的静态方法:
通过反编译的代码可以明确的看到, 非静态方法的局部变量表位置 0 存放的是 当前的this对象(注意奥:::是非静态方法奥)
总结:
如果是main方法的话,位置0存放的是args ,
如果是其他静态方法的话,就是从0开始存放的 ;
当然如果时非静态方法,位置0存放的就是当前this对象拉