javap与 i++,++i

javap 与 ++i,i++

javap命令

java命令:执行java源代码;
javac命令:将java源代码编译成.class文件
javap命令:对.class文件进行反编译,用于分解class文件.

既然我们对javap有了一定的了解,那我们就开始用它来解决一些实际的问题:
1.i++和++i的问题

执行 javap -verbose TestJavap.class命令 ,反编译结果为:

Code:
0: iconst_1//第一个局部变量 i
1: istore_1 //存储变量1到局部变量表中
2: iinc1, 1 //将局部变量1,也就是i,增加1,这个指令不会导致栈的变化,i此时变成2了
5: iconst_1//第二个局部变量2
6: istore_2//存储变量2到局部变量表
7: iinc2, 1//将局部变量2,也就是j,增加1,这个指令不会导致栈的变化,j此时变成2了
10: return

可以看出,++在前在后,在这段代码中,没有任何不同。

我们再看另一段代码:


反编译结果:

这里写图片描述

Code:
0: iconst_1 //对应第一个变量 int i
1: istore_1 //将局部变量1的值从操作数栈存储到局部变量表
2: iload_1 //将局部变量1加载到操作栈
3: iinc 1, 1 //局部变量1(即i)加1变为2,注意这时栈中仍然是1,没有改变
6: istore_1 //把栈顶的值放到局部变量1中,即i这时候由2变成了1
7: iconst_1 //对应第二个变量 j
8: istore_2 //将局部变量2的值从操作数栈存储到局部变量表
9: iinc 2, 1 //局部变量2(即j)加1变为2,注意这时栈中仍然是1,没有改变
12: iload_2 //把局部变量2(即j)的值放到栈顶,此时栈顶的值变为2
13: istore_2 //把栈顶的值放到局部变量2中,即j这时候真正由1变成了2
14: return

是否看明白了? 如果这个看明白了,那么下面的一个问题应该就是迎刃而解了:


2.i++与++i

m = m ++;这句话,java虚拟机执行时是这样的: m的值加了1,但这是栈中的值还是0, 马上栈中的值覆盖了m,即m变成0,因此不管循环多少次,m都等于0。

执行 javap命令:
这里写图片描述

  public TestJavap();
    descriptor: ()V
    flags: ACC_PUBLIC
    Code:
      stack=1, locals=1, args_size=1
         0: aload_0
         1: invokespecial #1                  // Method java/lang/Object."<init>
":()V
         4: return
      LineNumberTable:
        line 1: 0

  public static void main(java.lang.String[]);
    descriptor: ([Ljava/lang/String;)V
    flags: ACC_PUBLIC, ACC_STATIC
    Code:
      stack=2, locals=3, args_size=1
         0: iconst_0    //对应变量 m
         1: istore_1    //将变量m存储在局部变量表中
         2: iconst_0    //对应变量i
         3: istore_2    //将变量i存储在局部变量表中
         4: iload_2     //将变量i加载到操作栈中
         5: bipush        100   //将100加入到操作栈中
         7: if_icmpge     21    //将i大于等于100,跳到第21步
        10: iload_1     //将变量m加载到操作栈中
        11: iinc          1, 1      //将局部变量表中的m加1,即0+1;
        14: istore_1    //将栈中的变量值(此时为0) 存储到局部变量表中,所以在此之后局部变量表中的值又变成了0
        15: iinc          2, 1      //将将局部变量表中的i加1,即1+1;
        18: goto          4         //返回第四步,执行
        21: getstatic     #2                  // Field java/lang/System.out:Ljav
a/io/PrintStream;
        24: iload_1         //将局部变量表中的m值加载到操作栈中 ,此时为0
        25: invokevirtual #3                  // Method java/io/PrintStream.prin
tln:(I)V
        28: return

如果改为m = ++m; 程序运行结果就是100了。

/**
* @author zhangdi
*/
public class TestJavap {
        public static void main(String[] args) {
            mAdd();
            System.out.println("------------------");
            addM();
        }

        public static void mAdd() {
            int m = 0;
            for (int i = 0; i < 10; i++) {
                m = m++;
                System.out.println("m0:" + m);// m0:0
            }
            System.out.println("m:" + m);// m:0
        }

        public static void addM() {
            int m = 0;
            for (int i = 0; i < 10; i++) {
                m = ++m;
                System.out.println("m0:" + m);// m0:表现出的值为i+1
            }
            System.out.println("m:" + m);// m:10
        }
    }

参考自:https://blog.csdn.net/luckyp/article/details/4255353

posted @ 2018-07-29 22:38  XueXueLai  阅读(149)  评论(0编辑  收藏  举报