詳解java面試題中的i++和++i
代碼如下所示:
public class TestPlusPlus{ public static void main(String[] args){ int k = addAfterReturn(10); System.out.println(k); //輸出 10 int k1 = addbeforeReturn(10); System.out.println(k1); //輸出11 } public static int addbeforeReturn(int i){ return ++i; } public static int addAfterReturn(int i){ return i++; } }
我們從字節(jié)碼層面來(lái)看,
addAfterReturn的字節(jié)碼如下:
0: iload_0 1: iinc 0, 1 4: ireturn
很簡(jiǎn)單, iload_0 表示將局部變量表中索引為0的元素的值放到棧頂。所以講傳入的i的值放入到棧頂,其次 iinc 0,1 表示將局部變量表中
索引為0的元素進(jìn)行加1,最后是 ireturn 將棧頂?shù)膇nt值放入到調(diào)用者棧楨的棧中。(這里的棧都是操作數(shù)棧)
因此其實(shí) iinc 0,1 并沒(méi)有實(shí)際影響到返回的值,所以返回依舊是10.
同理addBeforeReturn字節(jié)碼如下:
0: iinc 0, 1 3: iload_0 4: ireturn
這里就留給大家自己分析,其實(shí)很明顯可以看出,這里是先進(jìn)行遞增然后才入棧了。所以返回的值其實(shí)是遞增了。
現(xiàn)在讓我們來(lái)做一道我自己改過(guò)的面試題。
public class TestPlusPlus2{ static{ x=5; int y = 10; } 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; } }
知道答案了嗎?運(yùn)行一下,正確答案是23.
如果你真正理解了上面的,這道題也很簡(jiǎn)單,首先是在執(zhí)行main之前,知道x,y的值,這里有點(diǎn)牽扯到類(lèi)加載機(jī)制,類(lèi)加載的準(zhǔn)備階段會(huì)為static變量賦值為該類(lèi)型的零值。int的話(huà)就對(duì)應(yīng)0嘍,在類(lèi)的初始化階段,根據(jù)代碼順序收集類(lèi)中的靜態(tài)塊和靜態(tài)變量賦值行為進(jìn)行生成 方法。因此該測(cè)驗(yàn)中,x = 5, y = 0;
繼續(xù)分析,在執(zhí)行myMethod之前執(zhí)行了 x--' ,所以x = 4,y = 0, myMethod 中 y = x++ + ++x;
x++在執(zhí)行后面的加操作之前是不會(huì)加1的,就跟 return i++ 在return之后才會(huì)進(jìn)行i++,(這里的return 你可以理解為上面的iload入棧操作,而不是iretrun這條指令),所以第一個(gè)加數(shù)為4,但是這里需要注意的是在執(zhí)行++x之前,遞增已經(jīng)對(duì)實(shí)際的x生效了,即x已經(jīng)為5了,所以第二個(gè)加數(shù)為 ++i 為6,從而y=10.
隨后就很簡(jiǎn)單了 x+y+ ++x 為 6+10+7 為23。幸運(yùn)的是如果你能看懂字節(jié)碼,你可以翻譯該代碼的字節(jié)碼,用字節(jié)碼來(lái)驗(yàn)算,也可以發(fā)現(xiàn)這樣的結(jié)果。
再來(lái)看一道例子 ,是群里大佬提供的。我覺(jué)得也很針對(duì)基礎(chǔ)。
public static int plus(int i){ try{ ++i; System.out.println("try"); return i++; }finally{ System.out.println("finally"); i++; //return i; } } public static void main(String[] args){ System.out.println(Test1.plus(5)); }
你能想到最后輸出的是多少嗎?如果去掉注釋呢,這也就是為什么finally塊里不要return的原因。因?yàn)闊o(wú)論對(duì)錯(cuò),可能永遠(yuǎn)都返回某個(gè)值。
總結(jié)
以上所述是小編給大家介紹的詳解java面試題中的i++和++i,希望對(duì)大家有所幫助,如果大家有任何疑問(wèn)請(qǐng)給我留言,小編會(huì)及時(shí)回復(fù)大家的。在此也非常感謝大家對(duì)腳本之家網(wǎng)站的支持!
相關(guān)文章
Spring不能注入Static變量的原因及Spring注入靜態(tài)變量
這篇文章主要介紹了Spring不能注入Static變量的原因及Spring注入靜態(tài)變量,需要的朋友可以參考下2016-01-01Spring實(shí)戰(zhàn)之緩存使用condition操作示例
這篇文章主要介紹了Spring實(shí)戰(zhàn)之緩存使用condition操作,結(jié)合實(shí)例形式分析了Spring緩存使用condition具體配置、屬性、領(lǐng)域模型等相關(guān)操作技巧與注意事項(xiàng),需要的朋友可以參考下2020-01-01spring schedule實(shí)現(xiàn)動(dòng)態(tài)配置執(zhí)行時(shí)間
這篇文章主要介紹了spring schedule實(shí)現(xiàn)動(dòng)態(tài)配置執(zhí)行時(shí)間,具有很好的參考價(jià)值,希望對(duì)大家有所幫助。如有錯(cuò)誤或未考慮完全的地方,望不吝賜教2021-11-11idea插件生成jpa實(shí)體類(lèi)的實(shí)現(xiàn)示例
本文主要介紹了idea插件生成jpa實(shí)體類(lèi)的實(shí)現(xiàn)示例,文中通過(guò)示例代碼介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友們下面隨著小編來(lái)一起學(xué)習(xí)學(xué)習(xí)吧2024-01-01因BigDecimal類(lèi)型數(shù)據(jù)引出的問(wèn)題詳析
Java在java.math包中提供的API類(lèi)BigDecimal,用來(lái)對(duì)超過(guò)16位有效位的數(shù)進(jìn)行精確的運(yùn)算,下面這篇文章主要給大家介紹了因BigDecimal類(lèi)型數(shù)據(jù)引出的問(wèn)題的相關(guān)資料,文中通過(guò)示例代碼介紹的非常詳細(xì),需要的朋友可以參考下2018-08-08使用RestTemplate 調(diào)用遠(yuǎn)程接口上傳文件方式
這篇文章主要介紹了使用RestTemplate 調(diào)用遠(yuǎn)程接口上傳文件方式,具有很好的參考價(jià)值,希望對(duì)大家有所幫助。如有錯(cuò)誤或未考慮完全的地方,望不吝賜教2021-09-09關(guān)于@ComponentScan注解的用法及作用說(shuō)明
這篇文章主要介紹了關(guān)于@ComponentScan注解的用法及作用說(shuō)明,具有很好的參考價(jià)值,希望對(duì)大家有所幫助,如有錯(cuò)誤或未考慮完全的地方,望不吝賜教2023-09-09通過(guò)實(shí)例深入學(xué)習(xí)Java的Struts框架中的OGNL表達(dá)式使用
這篇文章主要通過(guò)實(shí)例介紹了Java的Strus框架中的OGNL表達(dá)式使用,Struts框架是Java的SSH三大web開(kāi)發(fā)框架之一,需要的朋友可以參考下2015-11-11SpringBoot Maven打包失敗報(bào):class lombok.javac.apt.Lombo
最新項(xiàng)目部署的時(shí)候,出現(xiàn)了一個(gè)maven打包失敗的問(wèn)題,報(bào):class lombok.javac.apt.LombokProcessor錯(cuò)誤,所以本文給大家介紹了如何解決SpringBoot Maven 打包失敗:class lombok.javac.apt.LombokProcessor 錯(cuò)誤,需要的朋友可以參考下2023-12-12