在 Java 中,聲明類、變量和方法時,可使用關鍵字 final 來修飾。final 所修飾的數據具有“終態”的特征,表示“最終的”意思。具體規定如下:
* final 修飾的類不能被繼承。
* final 修飾的方法不能被子類重寫。
* final 修飾的變量(成員變量或局部變量)即成為常量,只能賦值一次。
* final 修飾的成員變量必須在聲明的同時賦值,如果在聲明的時候沒有賦值,那么只有 一次賦值的機會,而且只能在構造方法中顯式賦值,然后才能使用。
* final 修飾的局部變量可以只聲明不賦值,然后再進行一次性的賦值。
final 一般用于修飾那些通用性的功能、實現方式或取值不能隨意被改變的數據,以避免被誤用,例如實現數學三角方法、冪運算等功能的方法,以及數學常量π=3.141593、e=2.71828 等。
事實上,為確保終態性,提供了上述方法和常量的 java.lang.Math 類也已被定義為final 的。
需要注意的是,如果將引用類型(任何類的類型)的變量標記為 final,那么該變量不能指向任何其它對象。但可以改變對象的內容,因為只有引用本身是 final 的。
如果變量被標記為 final,其結果是使它成為常數。想改變 final 變量的值會導致一個編譯錯誤。下面是一個正確定義 final 變量的例子:
```
public class day05Stringclass2{
public static void main(String [] args){
public final int MAX_APP = 12; // 常量最好大寫
}
}
```
常量因為有 final 修飾,所以不能被繼承。
```
public final class Demo{
public static final int TOTAL_NUMBER = 5;
public int id;
public Demo() {
// 非法,對final變量TOTAL_NUMBER進行二次賦值了
// 因為++TOTAL_NUMBER相當于 TOTAL_NUMBER=TOTAL_NUMBER+1
id = ++TOTAL_NUMBER;
}
public static void main(String[] args) {
final Demo t = new Demo();
final int i = 10;
final int j;
j = 20;
j = 30; // 非法,對final變量進行二次賦值
}
}
```
final 也可以用來修飾類(放在 class 關鍵字前面),阻止該類再派生出子類,例如 Java.lang.String 就是一個 final 類。這樣做是出于安全原因,因為要保證一旦有字符串的引用,就必須是類 String 的字符串,而不是某個其它類的字符串(String 類可能被惡意繼承并篡改)。
方法也可以被 final 修飾,被 final 修飾的方法不能被覆蓋;變量也可以被 final 修飾,被 final 修飾的變量在創建對象以后就不允許改變它們的值了。一旦將一個類聲明為 final,那么該類包含的方法也將被隱式地聲明為 final,但是變量不是。
被 final 修飾的方法為靜態綁定,不會產生多態(動態綁定),程序在運行時不需要再檢索方法表,能夠提高代碼的執行效率。在Java中,被 static 或 private 修飾的方法會被隱式的聲明為 final,因為動態綁定沒有意義。
由于動態綁定會消耗資源并且很多時候沒有必要,所以有一些程序員認為:除非有足夠的理由使用多態性,否則應該將所有的方法都用 final 修飾。
這樣的認識未免有些偏激,因為 JVM 中的即時編譯器能夠實時監控程序的運行信息,可以準確的知道類之間的繼承關系。如果一個方法沒有被覆蓋并且很短,編譯器就能夠對它進行優化處理,這個過程為稱為內聯(inlining)。例如,內聯調用 e.getName() 將被替換為訪問 e.name 變量。這是一項很有意義的改進,這是由于CPU在處理調用方法的指令時,使用的分支轉移會擾亂預取指令的策略,所以,這被視為不受歡迎的。然而,如果 getName() 在另外一個類中被覆蓋,那么編譯器就無法知道覆蓋的代碼將會做什么操作,因此也就不能對它進行內聯處理了。
- Java自動化測試
- 第一章:Java:基礎內容
- 1.1:Java:Hello Word
- 1.2:Java:熱身
- 1.3:Java:注釋
- 1.4:Java:標識符
- 1.5:Java:常量
- 1.6:Java:基本數據類型
- 1.7:Java:引用類型
- 1.8:Java:String類
- 第二章:Java:運算符
- 2.1:Java:算數運算符
- 2.2:Java:關系運算符
- 2.3:Java:邏輯運算
- 2.4:Java:賦值運算符
- 2.5;Java:三元運算符
- 2.6:Java:位運算符
- 第三章:Java:循環控制語句
- 3.1:Java:for循環
- 3.2:Java:while循環
- 3.3:Java:switch
- 3.4:Java:if else
- 3.5:Java:練習題
- 第四章:Java:函數與全局/局部變量
- 4.1:Java:局部變量
- 4.2:Java:全局變量
- 第五章:Java:方法
- 5.1:Java:初識方法
- 5.2:Java:方法調用
- 5.3:Java:方法重載
- 5.4:Java:構造方法
- 5.5:Java:方法的注意事項
- 第六章:Java:面向對象
- 6.1:Java:小案例
- 6.2:Java:this 關鍵字
- 6.3:Java:super 關鍵字
- 6.4:Java:static 關鍵字
- 6.5:Java:final關鍵字
- 6.6:Java:instanceof 運算符
- 6.7:Java:面向對象之封裝
- 6.8:Java:面向對象之繼承
- 6.9:Java:面向對象之多態
- 第七章:Java:面向對象高級進階
- 7.1:Java:抽象類
- 7.2:Java:Java中String類
- 7.3:Java:interface接口
- 7.4:Java:ArrayList
- 7.5:Java:HashSet
- 7.6:Java:HashMap
- 7.7:Java:反射(reflection)
- 第八章:Java:日志以及異常捕獲
- 8.1:Java:log4j
- 8.2:Java:異常初識基礎
- 8.3:Java:未被捕獲的異常
- 8.4:Java:try和catch的使用
- 8.5:Java:多重catch語句的使用
- 8.6:Java:throws/throw 關鍵字
- 8.7:Java:finally關鍵字
- 8.8:Java:自定義異常
- 第九章:Java:xml and IO
- 9.1:Java:IO基本概念
- 9.2:java:properties
- 9.3:Java:xml基本介紹
- 9.4:Java:xml操作實例
- 第十章:Java:JDBC編程
- 10.1:Java:JDBC介紹
- 10.2:Java:JDBC查詢
- 10.3:Java:JDBC插入
- 10.4:Java:Batch
- 10.5:Java:JDBC連接池
- 第十一章:Java:TestNG
- 11.1:java:TestNG簡介
- 11.2:Java:TestNG小實例
- 11.3:Java:TestNG.xml文件配置
- 11.4:Java:TestNG基本注解
- 11.5:Java:TestNG注解代碼
- 11.6:Java:TestNG預期異常
- 11.7:Java:TestNG忽略測試
- 11.8:Java:TestNG超時測試
- 11.9:Java:TestNG分組測試