農(nóng)行一 面:說說 final,finally,finalize 的區(qū)別
在 Java中,final、finally和finalize是三個(gè)不同的關(guān)鍵字或方法,盡管它們的名字相似,但在功能和用途上卻有顯著的區(qū)別,這篇文章我們繼續(xù)來分析一篇農(nóng)行面試題目:說說 final、finally和finalize的區(qū)別。
final
final是一個(gè)保留關(guān)鍵字,用于修飾類、方法和變量。在 Java 中,final 關(guān)鍵字的主要作用是限制,并且確保某些行為不會被改變。主要表現(xiàn)如下:
- final變量:一旦被初始化就不能再被改變,即常量。當(dāng)聲明一個(gè)變量為final時(shí),必須在定義的時(shí)候進(jìn)行初始化,或者在構(gòu)造器中初始化,從而確保對應(yīng)的指針不會再指向其他對象。
- final方法:不能被子類重寫(override)。這樣確保方法行為保持一致,不被子類改變。
- final類:不能被繼承。通過將整個(gè)類聲明為final,防止其他類從它繼承。例如,String類就是一個(gè)final類,這樣可以保證字符串的不可變性。
下面給出幾個(gè) final的使用示例:
final變量:用于創(chuàng)建常量,在定義時(shí)必須初始化,減少錯(cuò)誤和提高易讀性:
final double PI = 3.14159;
final方法:確保方法的一致性和安全性,避免被子類篡改:
public final void display() {
System.out.println("This is a final method.");
}
final類:類被聲明為 final,意味著這個(gè)類不能被繼承。這確保了類的實(shí)現(xiàn)不能被其他類修改或擴(kuò)展。
public final class Constants {
public static final String APP_NAME = "FinalDemoApp";
}
finally
finally 是 Java 中的一個(gè)關(guān)鍵字,主要用于異常處理結(jié)構(gòu)中。它通常與 try 和 catch 塊聯(lián)用,是異常處理機(jī)制中一個(gè)非常重要的部分。finally 的執(zhí)行是幾乎保證的,無論是否發(fā)生異常,即便在 try 塊中有 return、break 或者 continue 語句,finally 塊仍然會執(zhí)行。但有極少數(shù)情況下可能不會執(zhí)行,例如:
- 如果在 try 或 catch 塊中調(diào)用了 System.exit() 方法,程序會退出,finally 塊不會執(zhí)行。
- 如果 JVM 出現(xiàn)了故障,比如操作系統(tǒng)層面的崩潰,這些都是程序無法控制的情況。
使用場景
在實(shí)際處理異常時(shí),finally 塊用于保證一些重要的清理操作,例如關(guān)閉資源,釋放鎖等,通常用于處理以下 3種場景:
(1) 資源管理
在編程實(shí)踐中,資源(如文件、數(shù)據(jù)庫連接、網(wǎng)絡(luò)連接等)的管理非常重要。finally 塊可以用來確保這些資源在使用后被正確關(guān)閉、釋放,避免資源泄漏。如下示例代碼:
FileInputStream fileInput = null;
try {
fileInput = new FileInputStream("example.txt");
// 處理文件
} catch (IOException e) {
e.printStackTrace();
} finally {
if (fileInput != null) {
try {
fileInput.close();
} catch (IOException e) {
e.printStackTrace();
}
}
}
(2) 恢復(fù)狀態(tài)
在異常處理過程中,系統(tǒng)可能會因?yàn)楫惓6幱谝环N不一致的狀態(tài)。finally 塊可以用來清理或者恢復(fù)這種狀態(tài),例如重置修改過的變量。如下示例代碼:
Lock lock = new ReentrantLock();
try {
lock.lock();
// 執(zhí)行一些可能拋出異常的操作
} finally {
lock.unlock(); // 確保鎖總是會被釋放
}
(3) 清除事務(wù)
在事務(wù)處理中,無論事務(wù)是否成功,finally 塊可以用來保證事務(wù)的閉合或清理等后續(xù)操作。例如在數(shù)據(jù)庫事務(wù)中,確保連接關(guān)閉。如下示例代碼:
Connection conn = null;
try {
conn = DriverManager.getConnection(DB_URL, USER, PASS);
conn.setAutoCommit(false);
// 執(zhí)行多步數(shù)據(jù)庫操作,可能拋出異常
conn.commit(); // 提交事務(wù)
} catch (SQLException e) {
if (conn != null) {
try {
conn.rollback(); // 回滾事務(wù)
} catch (SQLException ex) {
ex.printStackTrace();
}
}
e.printStackTrace();
} finally {
try {
if (conn != null) conn.close();
} catch (SQLException e) {
e.printStackTrace();
}
}
finalize
finalize() 是 JavaObject 類的一個(gè)方法,它允許對象在被垃圾收集器回收之前執(zhí)行清理操作。盡管在早期的 Java 版本中,finalize() 方法被設(shè)計(jì)用于釋放對象所持有的非 Java 語言的資源,例如關(guān)閉文件或網(wǎng)絡(luò)連接,但是在現(xiàn)代 Java 開發(fā)中,finalize() 已不再被推薦使用,其原因主要在于它的許多不確定性和低效性。Oracle 已建議開發(fā)者使用其他方式進(jìn)行資源管理,尤其是在 Java 9 及以后版本中,finalize() 已被標(biāo)記為過時(shí)(deprecated)。
1.finalize() 的基本原理
(1) 垃圾回收機(jī)制:
- 在 Java 中,垃圾回收器(Garbage Collector, GC)負(fù)責(zé)自動回收不再被引用的對象以釋放內(nèi)存。
- 當(dāng)垃圾回收器確定一個(gè)對象不再被引用時(shí),它會在該對象上調(diào)用 finalize() 方法,前提是該對象未被標(biāo)記為不可及狀態(tài)。
(2) 生命周期:
- 該方法可以被重寫用于執(zhí)行特定的清理任務(wù),比如釋放非托管資源。
- finalize() 方法只會被調(diào)用一次,即便對象在 finalize() 方法中重新被引用,這個(gè)方法也不會被再次調(diào)用。
2.使用 finalize() 的問題
不確定性:Java 的垃圾回收器無法保證 finalize() 方法會在對象死亡后立即執(zhí)行。執(zhí)行時(shí)間實(shí)際上是由 JVM 的垃圾收集來決定,這可能導(dǎo)致延遲清理和資源延遲釋放。
- 性能問題: 使用 finalize() 會增加 GC 的負(fù)擔(dān),因?yàn)閷ο笮枰欢啻螛?biāo)記和遍歷,導(dǎo)致一定的性能開銷。
- 錯(cuò)誤處理: 如果 finalize() 方法拋出異常,GC 只會忽略,異常不會傳播,這會導(dǎo)致難以調(diào)試的問題。
- 無法保證調(diào)用: 在程序正常終止之前,不一定會觸發(fā) GC,因此無法保護(hù)重要資源的釋放。
使用示例:
public class MyClass {
@Override
protected void finalize() throws Throwable {
// 執(zhí)行一些清理操作
}
}
三者對比
控制級別:
- final是編譯時(shí)屬性,用于類設(shè)計(jì)和限制,避免繼承和重寫。
- finally是運(yùn)行時(shí)捕獲異常處理后的保障機(jī)制,用于資源管理。
- finalize是執(zhí)行時(shí)的垃圾回收機(jī)制的一部分,但不再建議使用。
用途:
- final用于提供不可變性、繼承控制、重寫控制。
- finally用于異常處理中的資源清理。
- finalize過時(shí)的資源清理方法,替代為try-with-resources,try-with-resources極大提升了代碼的可讀性和可靠性。
總結(jié)
本文我們詳細(xì)地分析了final、finally和finalize以及它們之間的對比,實(shí)際上它們之間沒有什么直接關(guān)聯(lián),只是單詞的前 5個(gè)字符相同,所以在很多面試題中,經(jīng)常把它們放在一起進(jìn)行對比。對于這 3個(gè)關(guān)鍵字或者方法的建議是:
- 重點(diǎn)理解final關(guān)鍵字的使用
- 重點(diǎn)掌握finally在異常處理中的使用
- finalize方法已經(jīng)不再推薦,只需要了解