作者:恒如初见_193 | 来源:互联网 | 2023-08-26 18:19
最近看到个面试题:
public static void f4() {int j&#61;0;for(int i&#61;0;i<100;i&#43;&#43;){j &#61; j&#43;&#43;;}System.out.println(j);}
输出结果是0&#xff0c;如果换成j&#43;&#43;&#xff0c;那么输出100&#xff0c;这是为什么&#xff1f;
对于这种问题&#xff0c;其实有点经验的程序员都知道&#xff0c;前置&#43;&#43;和后置&#43;&#43;的运算区别&#xff1a;
1、&#43;&#43;j是先将变量的值加1&#xff0c;然后使用加1后的值参与运算
2、j&#43;&#43;是先使用该值参与运算&#xff0c;然后再将该值加1&#xff08;操作的是栈中的数据&#xff01;&#xff01;&#xff01;&#xff09;
话虽这么说&#xff0c;那怎么来证明上面两点呢&#xff1f;
万物皆有始有终&#xff0c;对于代码&#xff0c;只能从字节码的角度看看是不是能挖掘有价值的信息。
在讲字节码之前&#xff0c;先简单的了解下Java栈&#xff0c;在JVM中有这么一个数据结构叫Java栈&#xff0c;当线程启动的时候&#xff0c;会分配一块内存当做该线程的栈&#xff0c;每个栈由一系列的栈帧组成。每个栈帧对应一个方法&#xff0c;当线程执行方法时&#xff0c;就是栈帧出栈入栈的过程。
每个栈帧包含三部分数据&#xff1a;本地变量&#xff08;参数&#43;方法内的变量&#xff09;、操作数栈和其它数据&#xff0c;本文主要涉及本地变量和操作数栈。
&#xfeff;
先看看后置&#43;&#43;的实现&#xff1a;
public static void main(String[] args){int i&#61; 0;i &#61; i&#43;&#43;;
}
编译之后的字节码 javap -v Test.class > test.txt 如下&#xff1a;&#xfeff;
iconst_0 //把数值0 push到操作数栈
istore_1 // 把操作数栈写回到本地变量第2个位置
iload_1 // 把本地变量第2个位置的值push到操作数栈
iinc 1,1 //把本地变量表第2个位置加1 -----------在赋值给i之前 本地变量表&#xff0b;1&#xff01;&#xff01;&#xff01;&#xff01;
istore_1 // 把操作数据栈写回本地变量第2个位置
整个过程这样的&#xff1a;&#xfeff;
可以发现变量a在执行iinc 1,1的时候已经变成1了&#xff0c;但是istore_1又把变量a所在位置覆盖成了0&#xff0c;所以执行完i &#61; i&#43;&#43;&#xff0c;i还是原来那个值。
另外&#xff0c;来看下前置&#43;&#43;的实现&#xff1a;
&#xfeff;
字节码就不再解释了&#xff0c;整个过程实现如下&#xff1a;
&#xfeff;&#xfeff;和后置&#43;&#43;不同的地方在于&#xff0c;在变量进入操作数栈之前&#xff0c;就先执行了iinc指令&#xff0c;所以进入操作数的值是加1后的值&#xff0c;最后写回的值也是最新值。
转载自狼哥的博客 https://www.jianshu.com/p/7988e646a37e