代码如下所示:
1
2
3
4
5
6
7
8
9
10
11
12
13
14
|
public class TestPlusPlus{ public static void main(String[] args){ int k = addAfterReturn( 10 ); System.out.println(k); //输出 10 int k1 = addbeforeReturn( 10 ); System.out.println(k1); //输出11 } public static int addbeforeReturn( int i){ } public static int addAfterReturn( int i){ } } |
我们从字节码层面来看,
addAfterReturn的字节码如下:
1
2
3
|
0 : iload_0 1 : iinc 0 , 1 4 : ireturn |
很简单, iload_0 表示将局部变量表中索引为0的元素的值放到栈顶。所以讲传入的i的值放入到栈顶,其次 iinc 0,1 表示将局部变量表中
索引为0的元素进行加1,最后是 ireturn 将栈顶的int值放入到调用者栈桢的栈中。(这里的栈都是操作数栈)
因此其实 iinc 0,1 并没有实际影响到返回的值,所以返回依旧是10.
同理addBeforeReturn字节码如下:
1
2
3
|
0 : iinc 0 , 1 3 : iload_0 4 : ireturn |
这里就留给大家自己分析,其实很明显可以看出,这里是先进行递增然后才入栈了。所以返回的值其实是递增了。
现在让我们来做一道我自己改过的面试题。
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
|
public class TestPlusPlus2{ static { x= 5 ; int y = 10 ; } static int x,y; public static void main(String args[]){ x--; myMethod( ); System.out.println(x+y+ ++x); } public static void myMethod( ){ y=x++ + ++x; } } |
知道答案了吗?运行一下,正确答案是23.
如果你真正理解了上面的,这道题也很简单,首先是在执行main之前,知道x,y的值,这里有点牵扯到类加载机制,类加载的准备阶段会为static变量赋值为该类型的零值。int的话就对应0喽,在类的初始化阶段,根据代码顺序收集类中的静态块和静态变量赋值行为进行生成 方法。因此该测验中,x = 5, y = 0;
继续分析,在执行myMethod之前执行了 x--' ,所以x = 4,y = 0, myMethod 中 y = x++ + ++x;
x++在执行后面的加操作之前是不会加1的,就跟 return i++ 在return之后才会进行i++,(这里的return 你可以理解为上面的iload入栈操作,而不是iretrun这条指令),所以第一个加数为4,但是这里需要注意的是在执行++x之前,递增已经对实际的x生效了,即x已经为5了,所以第二个加数为 ++i 为6,从而y=10.
随后就很简单了 x+y+ ++x 为 6+10+7 为23。幸运的是如果你能看懂字节码,你可以翻译该代码的字节码,用字节码来验算,也可以发现这样的结果。
再来看一道例子 ,是群里大佬提供的。我觉得也很针对基础。
1
2
3
4
5
6
7
8
9
10
11
12
13
14
|
public static int plus( int i){ try { ++i; System.out.println( "try" ); return i++; } finally { System.out.println( "finally" ); i++; //return i; } } public static void main(String[] args){ System.out.println(Test1.plus( 5 )); } |
你能想到最后输出的是多少吗?如果去掉注释呢,这也就是为什么finally块里不要return的原因。因为无论对错,可能永远都返回某个值。
总结
以上所述是小编给大家介绍的详解java面试题中的i++和++i,希望对大家有所帮助,如果大家有任何疑问请给我留言,小编会及时回复大家的。在此也非常感谢大家对服务器之家网站的支持!