还剩2页未读,继续阅读
文本内容:
下面我将针对上次课程中,关于i++的问题给大家做个补充解释在上节课中,在讲到i++和++i的时候,给大家写了个例子Test{public classmai nStri ng[]args{pub Iicstat icvoidint i=10;i=20;inti=i++;System outpr int I ni;0Q原本在第5行,我是想写J=i++;来着,但是错写成i=i++;也正是由于这个语句,得出了令人意想不到的结果因为如果写成/i++;那么会输出尸11;而写成i=i++;则输出得到i=10;而如果同样的代码换成C/C++执行,则无论是i=i++还是j=i++;得到的结果是相同的,都是11,这是为什么呢?其实就i=i++;这个语句而言,其实很无聊,谁又会写这样一个语句呢写错除外1人,因为当你声明了i=0;后,要么你使用0,即直接使用i,要么你使用1,即用i++或者++i;没有必要写i=i++;所有这样的语句很无聊但是正是这么个无聊的语句,引发了一个关于编译器编译机制的大问题的讨论!为什么java得到户11,i=10;而C/C++得到的都是11呢?这是由于Java和C/C++采用的编译器不同,编译器不同,所以编译机制也不同,大家可以通过javap命令反编译出java虚拟机指令例如如下一段程序Test{public classmai nStri ng[]args{pub Iicstat icvoidint i=0;i++;System out.pr i ntIni;o通过运行命令javap-c Test得到如下结果:D:\javap-c TestCompiledfrom Test,java”public classTest extendsjava.lang.Object{public Test;Code:0:aload_01:invokespecial#1;//Method java/lang/Object.〃init〃V4:returnpublic staticvoid mainjava.lang.String[];Code:0:iconst_01:istore_l2:iload_l3:iinc1,16:istore_l7:getstatic#2;//Field java/lang/System.out:Ljava/io/PrintStream;10:iload_l11:invokevirtual#3;//Method java/io/PrintStream.printin:IV14:return对于上图中显示的结果大家可能读不懂,那么我在这里解释一下,从Code0开始:0:iconst_0〃将int类型常量0压入栈(栈顶)这里大家可能会犯嘀咕,int i=0;i不是变量么?i的确是变量,但赋值号后面的那个0是常量,可以这么说,对于基本数据类型的值(注意是值),都是常量这区别于引用类型,比如Student s=new Student();赋值号后面的是一■个对象,它是放在堆中的,而前面的这个s仍然是变量,是放在栈中的如图继续解释1:i store_1〃将int类型值存入局部变量12i load/〃从局部变量1中装载int类型值也就是说,这个时候把i变量中的值拿出来了,压入栈(栈顶),因为这里执行的是i++,也就是先使用后++,所有这句的目的是想把i在++之前的值先找个地方保存起来3:iinc1,1〃把常量值1加到局部变量1上这句话执行后i变量中存放的值就不是0了,而是1,注意刚才压入栈的那个值,也就是i在++之前的值,即栈顶的值仍然是06:istore_1〃把栈顶的int类型的值存入变量1中这句话就是用栈顶的0把i变量中的1给替换掉了所以最后得出了i=0的结果用几句通俗易懂的大白话讲一遍过程就是Java在执行变量=i++;在执行变量赋值语句(不是常量赋值,而是例如i二a;即赋值号后面有变量)之前,会先把i的值找个地方保存起来,然后直接在i变量的空间内对值+1,然后再把刚才保存起来的那个值赋给变量.那么清楚了上面的过程后,我们再来看看/i++;的JVM指令:E:\javap-c TestCompiledfrom Test,java”public classTest extendsjava.lang.Object{public Test0;Code:0:aload_0“〈1invokespecial#1;//Method java/lang/Object.init“V4:returnpublic staticvoid mainjava.lang.String[];Code:0:iconst_0」1is tore2:iconst_03istore_24:iload_l5:iinc1,18:istore_29:getstatic#2;//Field java/lang/System.out:Ljava/io/PrintStream,12:iload_l13:invokevirtual#3;//Method java/io/PrintStream.printin lV16:return过程是这样的0:把int类型的常量0压入栈顶1:把栈顶int类型的值存入变量1中(即给i赋值为0)2把int类型的常量0压入栈顶3:把栈顶int类型的值存入变量2中(即给J赋值为0)4:把变量1中的int类型的值装载到栈5把数值1加至〔J变量1中(给i变量的值加1)8把栈顶int类型的值存入变量2(即把0赋给j)所以,这时i的值为1,j的值为0所以大家在使用java中的i++时一定要注意,其他的没什么问题,只有i=i++;有问题,这和C/C++截然不同。