深入淺析Java中的final關(guān)鍵字
談到final關(guān)鍵字,想必很多人都不陌生,在使用匿名內(nèi)部類的時候可能會經(jīng)常用到final關(guān)鍵字。另外,Java中的String類就是一個final類,那么今天我們就來了解final這個關(guān)鍵字的用法。下面是本文的目錄大綱:
一.final關(guān)鍵字的基本用法
二.深入理解final關(guān)鍵字
若有不正之處,請多多諒解并歡迎指正。
final 對于常量來說,意味著值不能改變,例如 final int i=100。這個i的值永遠(yuǎn)都是100。但是對于變量來說又不一樣,只是標(biāo)識這個引用不可被改變,例如 final File f=new File("c:\\test.txt");那么這個f一定是不能被改變的,如果f本身有方法修改其中的成員變量,例如是否可讀,是允許修改的。形象的比喻:一個女子定義了一個final的老公,這個老公的職業(yè)和收入都是允許改變的,只是這個女人不會換老公而已。
一.final關(guān)鍵字的基本用法
在Java中,final關(guān)鍵字可以用來修飾類、方法和變量(包括成員變量和局部變量)。下面就從這三個方面來了解一下final關(guān)鍵字的基本用法。
1.修飾類
當(dāng)用final修飾一個類時,表明這個類不能被繼承。也就是說,如果一個類你永遠(yuǎn)不會讓他被繼承,就可以用final進(jìn)行修飾。final類中的成員變量可以根據(jù)需要設(shè)為final,但是要注意final類中的所有成員方法都會被隱式地指定為final方法。

在使用final修飾類的時候,要注意謹(jǐn)慎選擇,除非這個類真的在以后不會用來繼承或者出于安全的考慮,盡量不要將類設(shè)計為final類。
2.修飾方法
下面這段話摘自《Java編程思想》第四版第143頁:
“使用final方法的原因有兩個。第一個原因是把方法鎖定,以防任何繼承類修改它的含義;第二個原因是效率。在早期的Java實(shí)現(xiàn)版本中,會將final方法轉(zhuǎn)為內(nèi)嵌調(diào)用。但是如果方法過于龐大,可能看不到內(nèi)嵌調(diào)用帶來的任何性能提升。在最近的Java版本中,不需要使用final方法進(jìn)行這些優(yōu)化了?!?br />
因此,如果只有在想明確禁止 該方法在子類中被覆蓋的情況下才將方法設(shè)置為final的。
注:類的private方法會隱式地被指定為final方法。
3.修飾變量
修飾變量是final用得最多的地方,也是本文接下來要重點(diǎn)闡述的內(nèi)容。首先了解一下final變量的基本語法:
對于一個final變量,如果是基本數(shù)據(jù)類型的變量,則其數(shù)值一旦在初始化之后便不能更改;如果是引用類型的變量,則在對其初始化之后便不能再讓其指向另一個對象。
舉個例子:

上面的一段代碼中,對變量i和obj的重新賦值都報錯了。
二.深入理解final關(guān)鍵字
在了解了final關(guān)鍵字的基本用法之后,這一節(jié)我們來看一下final關(guān)鍵字容易混淆的地方。
1.類的final變量和普通變量有什么區(qū)別?
當(dāng)用final作用于類的成員變量時,成員變量(注意是類的成員變量,局部變量只需要保證在使用之前被初始化賦值即可)必須在定義時或者構(gòu)造器中進(jìn)行初始化賦值,而且final變量一旦被初始化賦值之后,就不能再被賦值了。
那么final變量和普通變量到底有何區(qū)別呢?下面請看一個例子:
public class Test {
public static void main(String[] args) {
String a = "hello2";
final String b = "hello";
String d = "hello";
String c = b + 2;
String e = d + 2;
System.out.println((a == c));
System.out.println((a == e));
}
}
true
false
大家可以先想一下這道題的輸出結(jié)果。為什么第一個比較結(jié)果為true,而第二個比較結(jié)果為fasle。這里面就是final變量和普通變量的區(qū)別了,當(dāng)final變量是基本數(shù)據(jù)類型以及String類型時,如果在編譯期間能知道它的確切值,則編譯器會把它當(dāng)做編譯期常量使用。也就是說在用到該final變量的地方,相當(dāng)于直接訪問的這個常量,不需要在運(yùn)行時確定。這種和C語言中的宏替換有點(diǎn)像。因此在上面的一段代碼中,由于變量b被final修飾,因此會被當(dāng)做編譯器常量,所以在使用到b的地方會直接將變量b 替換為它的 值。而對于變量d的訪問卻需要在運(yùn)行時通過鏈接來進(jìn)行。想必其中的區(qū)別大家應(yīng)該明白了,不過要注意,只有在編譯期間能確切知道final變量值的情況下,編譯器才會進(jìn)行這樣的優(yōu)化,比如下面的這段代碼就不會進(jìn)行優(yōu)化:
public class Test {
public static void main(String[] args) {
String a = "hello2";
final String b = getHello();
String c = b + 2;
System.out.println((a == c));
}
public static String getHello() {
return "hello";
}
}
這段代碼的輸出結(jié)果為false。
2.被final修飾的引用變量指向的對象內(nèi)容可變嗎?
在上面提到被final修飾的引用變量一旦初始化賦值之后就不能再指向其他的對象,那么該引用變量指向的對象的內(nèi)容可變嗎?看下面這個例子:
public class Test {
public static void main(String[] args) {
final MyClass myClass = new MyClass();
System.out.println(++myClass.i);
}
}
class MyClass {
public int i = 0;
}
這段代碼可以順利編譯通過并且有輸出結(jié)果,輸出結(jié)果為1。這說明引用變量被final修飾之后,雖然不能再指向其他對象,但是它指向的對象的內(nèi)容是可變的。
3.final和static
很多時候會容易把static和final關(guān)鍵字混淆,static作用于成員變量用來表示只保存一份副本,而final的作用是用來保證變量不可變。看下面這個例子:
public class Test {
public static void main(String[] args) {
MyClass myClass1 = new MyClass();
MyClass myClass2 = new MyClass();
System.out.println(myClass1.i);
System.out.println(myClass1.j);
System.out.println(myClass2.i);
System.out.println(myClass2.j);
}
}
class MyClass {
public final double i = Math.random();
public static double j = Math.random();
}
運(yùn)行這段代碼就會發(fā)現(xiàn),每次打印的兩個j值都是一樣的,而i的值卻是不同的。從這里就可以知道final和static變量的區(qū)別了。
4.匿名內(nèi)部類中使用的外部局部變量為什么只能是final變量?
這個問題請參見上一篇博文中《Java內(nèi)部類詳解》中的解釋,在此處不再贅述。
5.關(guān)于final參數(shù)的問題
關(guān)于網(wǎng)上流傳的”當(dāng)你在方法中不需要改變作為參數(shù)的對象變量時,明確使用final進(jìn)行聲明,會防止你無意的修改而影響到調(diào)用方法外的變量“這句話,我個人理解這樣說是不恰當(dāng)?shù)摹?br />
因?yàn)闊o論參數(shù)是基本數(shù)據(jù)類型的變量還是引用類型的變量,使用final聲明都不會達(dá)到上面所說的效果。
看這個例子就清楚了:

上面這段代碼好像讓人覺得用final修飾之后,就不能在方法中更改變量i的值了。殊不知,方法changeValue和main方法中的變量i根本就不是一個變量,因?yàn)閖ava參數(shù)傳遞采用的是值傳遞,對于基本類型的變量,相當(dāng)于直接將變量進(jìn)行了拷貝。所以即使沒有final修飾的情況下,在方法內(nèi)部改變了變量i的值也不會影響方法外的i。
再看下面這段代碼:
public class Test {
public static void main(String[] args) {
MyClass myClass = new MyClass();
StringBuffer buffer = new StringBuffer("hello");
myClass.changeValue(buffer);
System.out.println(buffer.toString());
}
}
class MyClass {
void changeValue(final StringBuffer buffer) {
buffer.append("world");
}
}
運(yùn)行這段代碼就會發(fā)現(xiàn)輸出結(jié)果為 helloworld。很顯然,用final進(jìn)行修飾并沒有阻止在changeValue中改變buffer指向的對象的內(nèi)容。有人說假如把final去掉了,萬一在changeValue中讓buffer指向了其他對象怎么辦。有這種想法的朋友可以自己動手寫代碼試一下這樣的結(jié)果是什么,如果把final去掉了,然后在changeValue中讓buffer指向了其他對象,也不會影響到main方法中的buffer,原因在于java采用的是值傳遞,對于引用變量,傳遞的是引用的值,也就是說讓實(shí)參和形參同時指向了同一個對象,因此讓形參重新指向另一個對象對實(shí)參并沒有任何影響。
以上所述是小編給大家分享的Java中的final關(guān)鍵字詳解,希望大家喜歡。
- JAVA中的final關(guān)鍵字用法實(shí)例詳解
- 詳解Java中的final關(guān)鍵字的使用
- 談?wù)凧ava中try-catch-finally中的return語句
- 簡單談?wù)刯ava的異常處理(Try Catch Finally)
- Java final static abstract關(guān)鍵字概述
- 簡單談?wù)刯ava中final,finally,finalize的區(qū)別
- Java中final關(guān)鍵字的用法總結(jié)
- Java中Finally關(guān)鍵字
- 深入解析Java編程中final關(guān)鍵字的作用
- 小議Java中final關(guān)鍵字使用時的注意點(diǎn)
相關(guān)文章
Java編程之多線程死鎖與線程間通信簡單實(shí)現(xiàn)代碼
這篇文章主要介紹了Java編程之多線程死鎖與線程間通信簡單實(shí)現(xiàn)代碼,具有一定參考價值,需要的朋友可以了解下。2017-10-10
SpringCloud+nacos部署在多ip環(huán)境下統(tǒng)一nacos服務(wù)注冊ip(親測有效)
在部署SpringCoud項(xiàng)目的時候分服務(wù)器部署注冊同一個nacos服務(wù),但是在服務(wù)器有多個ip存在的同時(內(nèi)外網(wǎng)),就會出現(xiàn)注冊服務(wù)ip不同的問題,導(dǎo)致一些接口無法連接訪問,經(jīng)過多次排查終于找到問題并找到解決方法,需要的朋友可以參考下2023-04-04
java警告:源發(fā)行版17 需要目標(biāo)發(fā)行版17問題及解決
文章介紹了如何解決項(xiàng)目JDK版本不一致的問題,包括修改Project Structure、Modules、Dependencies和Settings中的JDK版本,以及在pom.xml中指定JDK源版本2024-11-11
Activiti工作流學(xué)習(xí)筆記之自動生成28張數(shù)據(jù)庫表的底層原理解析
這篇文章主要介紹了Activiti工作流學(xué)習(xí)筆記之自動生成28張數(shù)據(jù)庫表的底層原理解析,本文給大家介紹的非常詳細(xì),對大家的學(xué)習(xí)或工作具有一定的參考借鑒價值,需要的朋友可以參考下2021-03-03
idea統(tǒng)計代碼行數(shù)Statistic的步驟詳解
這篇文章主要介紹了idea統(tǒng)計代碼行數(shù)Statistic的步驟詳解,本文通過使用Statistic插件操作的,通過圖文實(shí)例相結(jié)合給大家介紹的非常詳細(xì),對大家的學(xué)習(xí)或工作具有一定的參考借鑒價值,需要的朋友可以參考下2020-10-10

