[Java基礎]Finally

Duancf發表於2024-07-26

finally中的程式碼一定會執行嗎?

通常在面試中,只要是疑問句一般答案都是“否定”的,因為如果是“確定”和“正常”的,那面試官就沒有必要再問了嘛,而今天這道題的答案也是符合這個套路。

1.典型回答
正常執行的情況下,finally 中的程式碼是一定會執行的,但是,如果遇到以下異常情況,那麼 finally 中的程式碼就不會繼續執行了:

程式在 try 塊中遇到 System.exit() 方法,會立即終止程式的執行,這時 finally 塊中的程式碼不會被執行,例如以下程式碼:
public class FinallyExample {
public static void main(String[] args) {
try {
System.out.println("執行 try 程式碼.");
System.exit(0);
} finally {
System.out.println("執行 finally 程式碼.");
}
}
}
以上程式的執行結果如下:

在 try 快中遇到 Runtime.getRuntime().halt() 程式碼,強制終止正在執行的 JVM。與 System.exit()方法不同,此方法不會觸發 JVM 關閉序列。因此,當我們呼叫 halt 方法時,都不會執行關閉鉤子或終結器。實現程式碼如下:
public class FinallyExample {
public static void main(String[] args) {
try {
System.out.println("執行 try 程式碼.");
Runtime.getRuntime().halt(0);
} finally {
System.out.println("執行 finally 程式碼.");
}
}
}
以上程式的執行結果如下:

程式在 try 塊中遇到無限迴圈或者發生死鎖等情況時,程式可能無法正常跳出 try 塊,此時 finally 塊中的程式碼也不會被執行。
掉電問題,程式還沒有執行到 finally 就掉電了(停電了),那 finally 中的程式碼自然也不會執行。
JVM 異常崩潰問題導致程式不能繼續執行,那麼 finally 的程式碼也不會執行。
鉤子方法解釋
在程式設計中,鉤子方法(Hook Method)是一種由父類提供的空或預設實現的方法,子類可以選擇性地重寫或擴充套件該方法,以實現特定的行為或定製化邏輯。鉤子方法可以在父類中被呼叫,以提供一種可插拔的方式來影響父類的行為。
鉤子方法通常用於框架或模板方法設計模式中。框架提供一個骨架或模板,其中包含一些已經實現的方法及預留的鉤子方法。具體的子類可以透過重寫鉤子方法來插入定製邏輯,從而影響父類方法的實現方式。

2.考點分析
正常執行的情況下,finally 中的程式碼是一定會執行的,但是,如果遇到 System.exit() 方法或 Runtime.getRuntime().halt() 方法,或者是 try 中發生了死迴圈、死鎖,遇到了掉電、JVM 崩潰等問題,那麼 finally 中的程式碼也是不會執行的。

3.知識擴充套件
System.exit() 和 Runtime.getRuntime().halt() 都可以用於終止 Java 程式的執行,但它們之間有以下區別:

System.exit():來自 Java.lang.System 類的一個靜態方法,它接受一個整數引數作為退出狀態碼,通常非零值表示異常終止,使用零值表示正常終止。其中,最重要的是使用 exit() 方法,會執行 JVM 關閉鉤子或終結器。
Runtime.getRuntime().halt():來自 Runtime 類的一個例項方法,它接受一個整數引數作為退出狀態碼。其中退出狀態碼只是表示程式終止的原因,很少在程式終止時使用非零值。而使用 halt() 方法,不會執行 JVM 關閉鉤子或終結器。
例如以下程式碼,使用 exit() 方法會執行 JVM 關閉鉤子:

class ExitDemo {
// 註冊退出鉤子程式
static {
Runtime.getRuntime().addShutdownHook(new Thread(() -> {
System.out.println("執行 ShutdownHook 方法");
}));
}
public static void main(String[] args) {
try {
System.out.println("執行 try 程式碼。");
// 使用 System.exit() 退出程式
System.exit(0);
} finally {
System.out.println("執行 finally 程式碼。");
}
}
}
以上程式的執行結果如下:

而 halt() 退出的方法,並不會執行 JVM 關閉鉤子,示例程式碼如下:

class ExitDemo {

// 註冊退出鉤子程式
static {
    Runtime.getRuntime().addShutdownHook(new Thread(() -> {
        System.out.println("執行 ShutdownHook 方法");
    }));
}

public static void main(String[] args) {
    try {
        System.out.println("執行 try 程式碼。");
        // 使用 Runtime.getRuntime().halt() 退出程式
        Runtime.getRuntime().halt(0);
    } finally {
        System.out.println("執行 finally 程式碼。");
    }
}

}
以上程式的執行結果如下:

小結
正常執行的情況下,finally 中的程式碼是一定會執行的,但是,如果遇到 System.exit() 方法或 Runtime.getRuntime().halt() 方法,或者是 try 中發生了死迴圈、死鎖,遇到了掉電、JVM 崩潰等問題,finally 中的程式碼是不會執行的。而 exit() 方法會執行 JVM 關閉鉤子方法或終結器,但 halt() 方法並不會執行鉤子方法或終結器。