您的位置:首页 > 编程语言 > Java开发

java中i=i++问题分析

2016-07-09 18:57 309 查看
int i=0;
i=i++;
      结果i是多少?

      

      代码1:

public class Test{

    public static void main(String[] args){

         int i=0;

         i=i++;

         System.out.println(i);

    }

}

     结果i依然是0.

实际上

int i=0;
i=i++;等效于:

int temp = i;//这个temp就是i++这个表达式的值
i++; //i自增
i = temp;//最终,将表达式的值赋值给i

分析其反编译后的代码:   

public static void main(java.lang.String[]);

  Code:

   0:   iconst_0    
//0放到栈顶

   1:   istore_1   
//把栈顶的值保存到局部变量1,也就是i中

   2:   iload_1    
//把i的值放到栈顶,也就是说此时栈顶的值是0

   3:   iinc    1, 1 
//注意这个指令,把局部变量1,也就是i,增加1,这个指令不会导致栈的变化,也就是说局部变量1,即i此时为1了。

   6:   istore_1    
//把栈顶的值(0)保存到局部变量1,也就是让i为0了,所以最后i为0

   7:   getstatic   #2; //Field java/lang/System.out:Ljava/io/PrintStream;

   10:  iload_1

   11:  invokevirtual   #3; //Method java/io/PrintStream.println:(I)V

   14:  return

值得注意到是i被修改了两次,第一次是i++;i变为1,最后一次是i=0;所以结果i是0

代码2:

public class Test2{

    public static void main(String[] args){

         int i=0;

         int j=0;

         j=i++;

         System.out.println(i);

         System.out.println(j);

    }

}

这个结果肯定都知道,i是1,j是0.同样看反编译之后的代码: 

public static void main(java.lang.String[]);

  Code:

   0:   iconst_0

   1:   istore_1    
//i=0

   2:   iconst_0

   3:   istore_2    
//j=0

   4:   iload_1     
//把i的值放到栈顶,也就是说此时栈顶的值是0

   5:   iinc    1, 1 
//局部变量1加1,也就是让i++了,此时i已经是1了,上面说过,此指令不会导致栈变化

   8:   istore_2    
//把栈顶的值(注意是0)存入局部变量2,也就是j中,所以j=0

   9:   getstatic   #2; //Field java/lang/System.out:Ljava/io/PrintStream;

   12:  iload_1

   13:  invokevirtual   #3; //Method java/io/PrintStream.println:(I)V

   16:  getstatic   #2; //Field java/lang/System.out:Ljava/io/PrintStream;

   19:  iload_2

   20:  invokevirtual   #3; //Method java/io/PrintStream.println:(I)V

   23:  return

很明显可以看出,java是先把i的值取出来放到栈顶,我们可以认为是引入了第三个变量int k=i;然后i++,这时候i为1了,然后让j=k;也就是0.结论,i的++运算是在对j这个变量的赋值之前完成的。

代码3:

public class Test3{

    public static void main(String[] args){

         int i=0;

         int j=0;

         j=++i;

         System.out.println(i);

         System.out.println(j);

    }

}

结果大家也都知道,i=1,j=1

看操作过程: 

public static void main(java.lang.String[]);

  Code:

   0:   iconst_0    

   1:   istore_1   
//i=0

   2:   iconst_0

   3:   istore_2    
//j=0

   4:   iinc    1, 1  
//局部变量i加1,这时候i变成1了 。

   7:   iload_1    
//把i的值放到栈顶,栈顶的值是1

   8:   istore_2   
//j=1

   9:   getstatic   #2; //Field java/lang/System.out:Ljava/io/PrintStream;

   12:  iload_1

   13:  invokevirtual   #3; //Method java/io/PrintStream.println:(I)V

   16:  getstatic   #2; //Field java/lang/System.out:Ljava/io/PrintStream;

   19:  iload_2

   20:  invokevirtual   #3; //Method java/io/PrintStream.println:(I)V

   23:  return

对比代码2和代码3,关键的差别就是iload_1   个iinc这两条指令的位置变了。

 

 

当然:如果在C++中:上边的代码 运行结果必定是a=10.
内容来自用户分享和网络整理,不保证内容的准确性,如有侵权内容,可联系管理员处理 点击这里给我发消息
标签: