寫在前面
最近跳槽找工作的朋友確實不少,遇到的面試題也是千奇百怪,這不,一名讀者朋友面試時,被面試官問到了一個直擊靈魂的問題:if 語句執行完else語句真的不會再執行嗎?這個奇葩的問題把這名讀者問倒了!
問題分析
最近一名讀者留言說,自己出去面試被面試官的一道奇葩問題問倒了,這個問題就是:if語句執行完else語句真的不會再執行嗎?這名讀者確實不知道該如何回答這個問題。回去後,自己查閱了很多資料也沒弄明白這個問題!
想必很多讀者朋友遇到這種奇葩面試題時,多多少少都會覺得鬧心吧!不過,鬧心歸鬧心,問題還是要解決的。今天,我們就一起來剖析下這個奇葩的面試題。
從計算機底層原理來說,Java語句中的 if 指令和 else 指令分屬於兩個不同的邏輯分支,在同一段程式碼中,只要執行了if語句就不會執行else語句。所以,這個面試題的考點並不是讓你從計算機底層原理的角度去分析問題。既然不能從計算機底層原理去分析問題,那我們需要從哪裡入手分析呢?
沒錯,當然是從我們寫的程式入手了!那麼,問題來了,我們自己寫的程式貌似也沒有出現過執行完if語句後再執行else語句的情況呀!!別急,我們們繼續往下看。
實現程式
我們先來看一段程式碼,如下所示。
public class Test {
public static void main(String[] args) {
new Test().print(args==null || new Test() {{Test.main(null);}}.equals(null));
}
public void print(boolean flag){
if(flag){
System.out.println("我是if語句的分支");
}else{
System.out.println("我是else語句的分支");
}
}
}
在你的IDE中執行下這段程式,沒錯,輸出結果如下所示。
我是if語句的分支
我是else語句的分支
我去,竟然真的同時執行了if語句和else語句,這是怎麼回事呢?
程式碼分析
我們來看這段程式碼反編譯後的結果,如下所示。
public class Test {
public Test() {
}
public static void main(String[] args) {
(new Test()).print(args == null || (new Test() {
{
Test.main((String[])null);
}
}).equals((Object)null));
}
public void print(boolean flag) {
if (flag) {
System.out.println("我是if語句的分支");
} else {
System.out.println("我是else語句的分支");
}
}
}
看到這裡,有木有一種恍然大悟的感覺呢?沒錯,上述的程式在本質上,main方法執行了兩次。為什麼會是執行了兩次呢?原因就在main方法中呼叫print()方法時,傳遞的引數上。所以,我們先來看看呼叫print()方法傳遞的引數,如下所示。
args == null || (new Test() {
{
Test.main((String[])null);
}
}).equals((Object)null)
可以看到,呼叫print()方法傳遞的引數中,args == null為true,執行print()方法的if語句,這點不難理解。接下來就是要重點理解下面的程式碼片段了。
(new Test() {
{
Test.main((String[])null);
}
}).equals((Object)null)
這段程式碼是什麼意思呢?首先,這段程式碼再次建立了一個Test類的物件例項,並在程式碼塊中呼叫了Test類的main()方法,此時,由於Test類的物件例項不為空,所以,equals((Object)null)會返回false。此時,再次執行print()方法時,傳遞的flag為false,執行了else語句的邏輯。
是不是很神奇呢?所以,從現在開始,你要轉變你的觀念,這告訴我們:任何權威都不是絕對的,你要做的就是要敢於挑戰權威,指出他們不對的地方!
寫在最後
如果覺得文章對你有點幫助,請微信搜尋並關注「 冰河技術 」微信公眾號,跟冰河學習高併發程式設計技術。
最後,附上併發程式設計需要掌握的核心技能知識圖,祝大家在學習併發程式設計時,少走彎路。