其實大家只要記住一句話就可以了,前++是先自加再使用而后++是先使用再自加疾渣!
前++和后++總結(jié):其實大家只要記住一句話就可以了篡诽,前++是先自加再使用而后++是先使用再自加!
總結(jié):
_x 負(fù)責(zé)中間變量保存i的值榴捡,最后一步都是_x賦值給等號左邊的變量
++在前的時候杈女,先自增,在保存值到_x,最后_x 的值賦值給等號右邊的變量
++在后的時候吊圾,_x先保存值达椰,在自增,然后把_x的值賦值給等號右邊的變量
真實面試題與解析
看下面一段程序项乒,相信每一個標(biāo)準(zhǔn)的程序員都做過:
package hello_java;
public class Test {
public static void main(String[] args) {
int i = 0;
i = i ++;
System.out.println(i);
}
}
這段代碼的運行結(jié)果啰劲,估計多數(shù)人都背住了,但是底層究竟是怎么運行的檀何,可能就蝇裤。趁尼。。之前學(xué)習(xí)的時候猖辫,完全不認(rèn)為這是一個問題酥泞,但是有一天當(dāng)別人問我的時候,我竟發(fā)現(xiàn)自己只是記住了答案啃憎,隱約記得解釋芝囤,但是就是說不出來,今天要pass掉這個問題辛萍。先看答案:
0
答案為什么是0悯姊?最底層的原因是 i = i++操作并不是原子性,而是分步進行的贩毕,如果你不能理解什么是原子性悯许,可以先看這個操作
1 + 1 + 1 = ?
我想你大概率會直接得到答案3辉阶,但是對于一個小朋友來講先壕,他要先計算1 + 1 得到2之后再和1相加,得到最終的結(jié)果3 谆甜,成年人雖然能夠直接得到3這個答案垃僚,但是實際上我們中間是有一個2的,1 + 1 是原子性的规辱,但是1 + 1 + 1 就不在是原子性的了谆棺。
對于i = i ++ 其實是存在一個中間變量比如 _x ,然后在將 _x 的值賦值給 i ,得到最后的結(jié)果:
_x = i ++ ; i 的值賦值給 _x ,然后 i 自增為 1 ,_x 為 0
i = _x ; i 的值被覆蓋為 0
所以最終的結(jié)果是 0
= 的作用是將 = 右邊的計算結(jié)果賦值給左邊
最后罕袋,將下面的代碼和上面的代碼做一個比較:
package hello_java;
public class Test {
public static void main(String[] args) {
int i = 0;
int x ;
x = i ++;
System.out.println(x);//0
System.out.println(i);//1
}
}
public class Test {
public static void main(String[] args) {
int i = 0;
i = ++i;
System.out.println(i); //1
/*
i自增為1改淑,_x 保存i的值,最后_x賦值給等號左邊的變量
*/
}
}
注意: 一般而言賦值操作是原子性的浴讯,但是排除 long 和 double朵夏,為什么?因為他們倆都是64位的兰珍,在進行分配內(nèi)存的時候侍郭,不會直接分配那么大的內(nèi)存询吴,所以分兩次分內(nèi)存掠河,一次分配32 位。
大招:
最牛叉的是編譯一下猛计,看看class文件是怎么回事唠摹,到底是怎么走的:
i= i ++;
int i = 0;
byte var10000 = i;
int var2 = i + 1;
i = var10000;
System.out.println(i);
i = ++ i奉瘤;
int i = 0;
int i = i + 1;
System.out.println(i);
這還不是最底層勾拉,編譯之后煮甥,我們查看字節(jié)碼文件,這需要匯編語言的基礎(chǔ)藕赞,還好我大學(xué)的時候?qū)W習(xí)過匯編語言
總結(jié):
_x 負(fù)責(zé)中間變量保存i的值成肘,最后一步都是_x賦值給等號左邊的變量
++在前的時候,先自增斧蜕,在保存值到_x,最后_x 的值賦值給等號右邊的變量