java

2.深入剖析Java中i++和++i

2018-02-28  本文已影响16人  南城刀

下面程序输出的结果是多少?

public class  Test  {
    static {
          int x = 5;
    }

    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;
  }
}

解析:首先,从题目中代码来看,我们不难看出这一道题考察的是静态块和 i++ /++i的问题。所以我们看到这个可以想到的是:

  i++   先运算,后赋值
  ++i   先赋值,后运算

但是,仅仅知道上面这些的话,而没有真正理解的话。如果碰到一些情况还是会弄混淆的。例如:

(1) public static void main ( String[]  args ) {
       int  i = 0;
       System.out.println(i++);
       int j = 0;
        System.out.println(++j);
 }
如果记住上面的规则,那么这个输出的答案可能会答错,认为是1和0。正确答案是0和1;

如果将上面的例子稍微修改一下:

(2)public static void main ( String[]  args ) {
       int  i = 0;
       i++;
       System.out.println(i);
       int j = 0;
       ++j;
        System.out.println(j);
 };
 正确答案是1和1;

因为看的很多视频课程将都是告诉你记住上面规则,但是有时候你稍微不注意就会出错的,他的规则到底是什么样的?
借助于百度找到关于i++和++i底层的解释是这样的:

i++:Fetch i,copy i,increment i,return copy; 
++i:Fetch i,increment i,return i;

对译

i++:取出i,复制i,增加i,返回副本; 
++i:取出i,增加i,返回i;  

从上面我们不难看出:i++是一个表达式,是有返回值的,它的返回值就是 i 自增前的值,java对于自增是这样处理的:先把i的值(注意是值,不是引用)拷贝到一个临时变量区,然后对i变量加1,最后返回临时变量区的值。程序第一次循环时的详细处理步骤如下:

步骤1:JVM把 i 的值拷贝到临时变量区 
步骤2:i 值加1,这时候 i 的值是1 
步骤3:返回临时变量区的值,0 
步骤4:返回值赋值给 i,此时 i 值被重置成0.

所以(1)中输出的(i++)可以看成是一个表达式们,一个整体,输出的是临时变量,那为什么(2)中的 将 i++拿到外面,就不一样了呢?拿到外面的话,我们可以这样分析,首先它不是一个表达式了,而是运算了,可以看成是( i = i +1),按照上面分析,其实 i = 0 + 1,所以(2)中 i 输出的是 1.

对题目中的代码解析如下:

public class Test {
    static { int x = 5;}//Java类中的静态代码块,是在类第一次载入JVM时运行,但是由于是局部变量,x=5 不影响后面的值。
    static int x,y;//在初始化变量的时候  x=0,y=0;
    public static void main {
            x--;
            System.out.println(x);//步骤1:在运行myMethod() ;  之前,x 是 -1,接下来开始调用myMethod()函数
    }
      public static void myMethod(){
          y = x++ + ++x;
          System.out.println(y);//步骤2:进入myMethod()方法 运行 y = (x++)+  (++x)后y = 0;
          System.out.println(x);//步骤3: 此时 x=1
      }
}

所以最终的答案是2;

上一篇下一篇

猜你喜欢

热点阅读