天天看點

Java基礎(五) final關鍵字淺析

本文主要對final關鍵字的原理進行了講解,同時對其基本用法進行了說明,包括final修飾的類,final修飾的方法和final修飾的變量,另外文中String變量通過==比較隻是為了更加清晰的說明final原理,實際應用場景比較的時候還是用equals()方法,final也經常和static配合使用作為“全局常量”

前面在講解String時提到了final關鍵字,本文将對final關鍵字進行解析。

static和final是兩個我們必須掌握的關鍵字。不同于其他關鍵字,他們都有多種用法,而且在一定環境下使用,可以提高程式的運作性能,優化程式的結構。下面我們來了解一下final關鍵字及其用法。

final從總體上來說是“不可變的”,可用于修改類、方法、變量。

一. final類

final修飾的類,該類不能被繼承。當你确認一個類永遠不會被繼承或不想被繼承,那麼就可以用final修飾。

Java基礎(五) final關鍵字淺析

同樣,對于接口(interface)和抽象類(abstract Class),其本就是為了“多态”而設計,自然無法用final關鍵字修飾

Java基礎(五) final關鍵字淺析
Java基礎(五) final關鍵字淺析

final類中的成員方法預設也被隐式指定為final方法。

二. final方法

final修飾的方法不可被重寫。

例子:

/**
 * 父類
 * @author LH
 */
public class FinalDemo1 {
    public final void test() {
        
    }
}
           
Java基礎(五) final關鍵字淺析

三. final變量

final變量包括成員變量和局部變量。變量類型包括基本資料類型、對象。

通過final修飾局部基本類型變量(及其包裝類),數值一經初始化(可以定義時初始化,也可以在使用前初始化)不可改變。如:

final int a = 0;
a = 1;//報錯
final int b;
b = 1;//編譯通過
           

通過final修飾局部引用類型變量時,其引用的對象(記憶體位址)(可以定義時初始化,也可以在使用前初始化)不可改變,但對象中存放的資料可以改變

public static void main(String[] args) {
	final String str1 = "helloWorld";
	str1 = "helloChina";//編譯出錯,String的不可變性,此處傳回的是新的對象引用。

	final StringBuilder sb = new StringBuilder("hello");
	sb.append("world");//編譯通過

	sb = new StringBuilder("China");//編譯出錯
}
           

final修飾的成員變量必須在定義的時候直接初始化,否則會編譯出錯

public class FinalDemo1 {
	public final int age;//final修飾的基本類型,編譯出錯
	public final int age1 = 20;//final修飾的基本類型,編譯通過
    public final StringBuilder address;// final修飾的引用類型,編譯出錯
    public final StringBuilder address1 = new StringBuilder("中國");//final修飾的引用類型,編譯通過
}
           

那麼final變量與普通變量之間到底有何差別,看下面的例子

public static void main(String[] args) {
    String str0 = "helloWorldChina";
    String str1 = "helloWorld";
    String str3 = str1 + "China";
    System.out.println(str0 == str3);//false
    
    final String str2 = "helloWorld";
    String str4 = str2 + "China";
    System.out.println(str0 == str4);//true
    
    final String str5;
    str5 = "helloWorld";
    String str6 = str5 + "China";
    System.out.println(str0 == str6);//false
}
           

str0 == str3運作結果為false,這是因為通過“+”生成了一個新的字元串對象,傳回的引用位址和str0不再一樣,這在《Java基礎(三) String深度解析》中有講解。

那麼str0 == str4的執行結果為什麼是true?

通過final修飾的變量,如果在編譯期都可以知道确切值(定義變量的時候就初始化),那麼在編譯器會将其當做常量使用,所有用到該變量的地方就相當于直接使用該常量,String str4 = str2 + "China" 在編譯期間都已經合并處理成String str4 = "helloWorldChina",是以str0與str4引用了常量池中同一個字元串字面量的位址,故而結果為true。

而str0 == str6的執行結果為false也很好了解

str5在編譯期并不知道确切值,而是在使用之前才進行初始化,是以編譯器無法事先進行合并處理,str6通過“+”生成了一個新的字元串對象,傳回的引用位址和str0也不再一樣。

而針對基本資料類型來說定義為final變量與普通變量,比較結果來說并無差異

public static void testint(){
	int int0 = 8;    
	final int int1;    
	int1 = 4;    
	int int2 = int1 + 4;    
	System.out.println(int2 == int0);//true
}
           

因為基本資料類型并不存在引用傳遞的概念,基本類型變量也是字面常量,是以對基本類型的操作都是直接對值的操作,和引用不一樣,比較的并非位址。

四. 總結

本文主要對final關鍵字的原理進行了講解,同時對其基本用法進行了說明,包括final修飾的類,final修飾的方法和final修飾的變量,另外文中String變量通過==比較隻是為了更加清晰的說明final原理,實際應用場景比較的時候還是用equals()方法,final也經常和static配合使用作為“全局常量”,若有不對之處,請批評指正,望共同進步,謝謝!