寫在前面
最近跳槽找工作的朋友確實不少,遇到的面試題也是千奇百怪,這不,一名讀者朋友面試時,被面試官問到了一個直擊靈魂的問題: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語句的邏輯。
是不是很神奇呢?所以,從現在開始,你要轉變你的觀念,這告訴我們:任何權威都不是絕對的,你要做的就是要敢於挑戰權威,指出他們不對的地方!
寫在最后
如果覺得文章對你有點幫助,請微信搜索並關注「 冰河技術 」微信公眾號,跟冰河學習高並發編程技術。
最后,附上並發編程需要掌握的核心技能知識圖,祝大家在學習並發編程時,少走彎路。