final、finally与finalize的区别
生活随笔
收集整理的這篇文章主要介紹了
final、finally与finalize的区别
小編覺得挺不錯的,現(xiàn)在分享給大家,幫大家做個參考.
1. final
在java中,final可以用來修飾類,方法和變量(成員變量或局部變量)。1.1 修飾類
當用final修飾類的時,表明該類不能被其他類所繼承。當我們需要讓一個類永遠不被繼承,此時就可以
用final修飾,但要注意:
final類中所有的成員方法都會隱式的定義為final方法。1.2 修飾方法
使用final方法的原因主要有兩個:
(1) 把方法鎖定,以防止繼承類對其進行更改。
(2) 效率,在早期的java版本中,會將final方法轉為內嵌調用。但若方法過于龐大,可能在性能上不會有多大
提升。因此在最近版本中,不需要final方法進行這些優(yōu)化了。final方法意味著“最后的、最終的”含義,即此方法不能被重寫。注意:若父類中final方法的訪問權限為private,將導致子類中不能直接繼承該方法,因此,此時可以在
子類中定義相同方法名的函數(shù),此時不會與重寫final的矛盾,而是在子類中重新地定義了新方法。class A{private final void getName(){}
}public class B extends A{public void getName(){}public static void main(String[]args){System.out.println("OK");}
}1.3 修飾變量
final成員變量表示常量,只能被賦值一次,賦值后其值不再改變。類似于C++中的const。當final修飾一個基本數(shù)據(jù)類型時,表示該基本數(shù)據(jù)類型的值一旦在初始化后便不能發(fā)生變化;如果final
修飾一個引用類型時,則在對其初始化之后便不能再讓其指向其他對象了,但該引用所指向的對象的內容是
可以發(fā)生變化的。本質上是一回事,因為引用的值是一個地址,final要求值,即地址的值不發(fā)生變化。 final修飾一個成員變量(屬性),必須要顯示初始化。這里有兩種初始化方式,一種是在變量聲明的時候
初始化;第二種方法是在聲明變量的時候不賦初值,但是要在這個變量所在的類的所有的構造函數(shù)中對這個
變量賦初值。當函數(shù)的參數(shù)類型聲明為final時,說明該參數(shù)是只讀型的。即你可以讀取使用該參數(shù),但是無法改變該
參數(shù)的值。在java中,String被設計成final類,那為什么平時使用時,String的值可以被改變呢?字符串常量池是java堆內存中一個特殊的存儲區(qū)域,當我們建立一個String對象時,假設常量池不存在該
字符串,則創(chuàng)建一個,若存在則直接引用已經(jīng)存在的字符串。當我們對String對象值改變的時候,
例如 String a="A"; a="B" 。a是String對象的一個引用(我們這里所說的String對象其實是指字符串
常量),當a=“B”執(zhí)行時,并不是原本String對象("A")發(fā)生改變,而是創(chuàng)建一個新的對象("B"),令a
引用它。2. finally
finally作為異常處理的一部分,它只能用在try/catch語句中,并且附帶一個語句塊,表示這段語句最終
一定會被執(zhí)行(不管有沒有拋出異常),經(jīng)常被用在需要釋放資源的情況下。(×)(這句話其實存在一定的
問題)很多人都認為finally語句塊一定會執(zhí)行,但真的是這樣么?答案是否定的,例如下面這個例子:只有與finally對應的try語句塊得到執(zhí)行的情況下,finally語句塊才會執(zhí)行。以上兩種情況在執(zhí)行try
語句塊之前已經(jīng)返回或拋出異常,所以try對應的finally語句并沒有執(zhí)行。但是,在某些情況下,即使try語句執(zhí)行了,finally語句也不一定執(zhí)行。例如以下情況:finally 語句塊還是沒有執(zhí)行,為什么呢?因為我們在 try 語句塊中執(zhí)行了 System.exit (0) 語句,
終止了 Java 虛擬機的運行。那有人說了,在一般的 Java 應用中基本上是不會調用這個System.exit(0)
方法的。OK !沒有問題,我們不調用 System.exit(0) 這個方法,那么 finally語句塊就一定會執(zhí)行嗎?再一次讓大家失望了,答案還是否定的。當一個線程在執(zhí)行 try 語句塊或者 catch 語句塊時
被打斷(interrupted)或者被終止(killed),與其相對應的 finally 語句塊可能不會執(zhí)行。
還有更極端的情況,就是在線程運行 try 語句塊或者 catch 語句塊時,突然死機或者斷電,finally
語句塊肯定不會執(zhí)行了。可能有人認為死機、斷電這些理由有些強詞奪理,沒有關系,我們只是為了說明
這個問題。易錯點
在try-catch-finally語句中執(zhí)行return語句。我們看如下代碼:
首先finally語句在改代碼中一定會執(zhí)行,從運行結果來看,每次return的結果都是4(即finally語句),
仿佛其他return語句被屏蔽掉了。
事實也確實如此,因為finally用法特殊,所以會撤銷之前的return語句,繼續(xù)執(zhí)行最后的finally塊中的
代碼。 3. finalize
finalize()是在java.lang.Object里定義的,也就是說每一個對象都有這么個方法。這個方法在gc啟動,
該對象被回收的時候被調用。其實gc可以回收大部分的對象(凡是new出來的對象,gc都能搞定,一般情況下
我們又不會用new以外的方式去創(chuàng)建對象),所以一般是不需要程序員去實現(xiàn)finalize的。
特殊情況下,需要程序員實現(xiàn)finalize,當對象被回收的時候釋放一些資源,比如:一個socket鏈接,
在對象初始化時創(chuàng)建,整個生命周期內有效,那么就需要實現(xiàn)finalize,關閉這個鏈接。 使用finalize還需要注意一個事,調用super.finalize();一個對象的finalize()方法只會被調用一次,而且finalize()被調用不意味著gc會立即回收該對象,所以
有可能調用finalize()后,該對象又不需要被回收了,然后到了真正要被回收的時候,因為前面調用過
一次,所以不會調用finalize(),產(chǎn)生問題。 所以,推薦不要使用finalize()方法,它跟析構函數(shù)不一樣。
?
總結
以上是生活随笔為你收集整理的final、finally与finalize的区别的全部內容,希望文章能夠幫你解決所遇到的問題。
- 上一篇: 多线程——线程的生命周期
- 下一篇: 什么是单例模式?