天天看點

【Java】面試官靈魂拷問: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語句的分支      
【Java】面試官靈魂拷問:if語句執行完else語句真的不會再執行嗎?

我去,竟然真的同時執行了if語句和else語句,這是怎麼回事呢?

【Java】面試官靈魂拷問: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語句的邏輯。

【Java】面試官靈魂拷問:if語句執行完else語句真的不會再執行嗎?

是不是很神奇呢?是以,從現在開始,你要轉變你的觀念,這告訴我們:任何權威都不是絕對的,你要做的就是要敢于挑戰權威,指出他們不對的地方!