在Java 中,final關(guān)鍵字可以修飾 類瘪弓,方法,和變量(包含成員變量和局部變量)禽最。
final 類
當(dāng)用final修飾一個(gè)類時(shí)腺怯,表明這個(gè)類不能被繼承。也就是說川无,如果一個(gè)類你永遠(yuǎn)不會(huì)讓他被繼承呛占,就可以用final進(jìn)行修飾。final類中的成員變量可以根據(jù)需要設(shè)為final懦趋,但是要注意final類中的所有成員方法都會(huì)被隱式地指定為final方法晾虑。
String 類就是final類。
final 方法
不允許子類重寫方法可以將方法聲明為final仅叫。
類的private 方法會(huì)隱式的指定為final帜篇。
final 變量
對(duì)于一個(gè)final變量,如果是基本數(shù)據(jù)類型的變量诫咱,則其數(shù)值一旦在初始化之后便不能更改笙隙;如果是引用類型的變量,則在對(duì)其初始化之后便不能再讓其指向另一個(gè)對(duì)象坎缭。
當(dāng)用final作用于類的成員變量時(shí)竟痰,成員變量(注意是類的成員變量,局部變量只需要保證在使用之前被初始化賦值即可)必須在定義時(shí)或者構(gòu)造器中進(jìn)行初始化賦值幻锁,而且final變量一旦被初始化賦值之后凯亮,就不能再被賦值了。
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));
}
}
大家可以先想一下這道題的輸出結(jié)果哄尔。為什么第一個(gè)比較結(jié)果為true假消,而第二個(gè)比較結(jié)果為fasle。這里面就是final變量和普通變量的區(qū)別了岭接,當(dāng)final變量是基本數(shù)據(jù)類型以及String類型時(shí)富拗,如果在編譯期間能知道它的確切值臼予,則編譯器會(huì)把它當(dāng)做編譯期常量使用。也就是說在用到該final變量的地方啃沪,相當(dāng)于直接訪問的這個(gè)常量粘拾,不需要在運(yùn)行時(shí)確定。這種和C語言中的宏替換有點(diǎn)像创千。因此在上面的一段代碼中缰雇,由于變量b被final修飾,因此會(huì)被當(dāng)做編譯器常量追驴,所以在使用到b的地方會(huì)直接將變量b 替換為它的 值械哟。而對(duì)于變量d的訪問卻需要在運(yùn)行時(shí)通過鏈接來進(jìn)行。想必其中的區(qū)別大家應(yīng)該明白了殿雪,不過要注意暇咆,只有在編譯期間能確切知道final變量值的情況下,編譯器才會(huì)進(jìn)行這樣的優(yōu)化丙曙,比如下面的這段代碼就不會(huì)進(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";
}
}
這段代碼的輸出false
關(guān)于final參數(shù)
上面這段代碼好像讓人覺得用final修飾之后爸业,就不能在方法中更改變量i的值了。殊不知亏镰,方法changeValue和main方法中的變量i根本就不是一個(gè)變量扯旷,因?yàn)閖ava參數(shù)傳遞采用的是值傳遞,對(duì)于基本類型的變量拆挥,相當(dāng)于直接將變量進(jìn)行了拷貝薄霜。所以即使沒有final修飾的情況下,在方法內(nèi)部改變了變量i的值也不會(huì)影響方法外的i纸兔。
public class Test {
publicstaticvoidmain(String[] args)? {
MyClass myClass =new MyClass();
StringBuffer buffer =new StringBuffer("hello");
myClass.changeValue(buffer);
System.out.println(buffer.toString());
}
}
class MyClass {
void change Value(final StringBuffer buffer) {
buffer.append("world");
}
}
運(yùn)行這段代碼就會(huì)發(fā)現(xiàn)輸出結(jié)果為 helloworld惰瓜。很顯然,用final進(jìn)行修飾并沒有阻止在changeValue中改變buffer指向的對(duì)象的內(nèi)容汉矿。有人說假如把final去掉了崎坊,萬一在changeValue中讓buffer指向了其他對(duì)象怎么辦。有這種想法的朋友可以自己動(dòng)手寫代碼試一下這樣的結(jié)果是什么洲拇,如果把final去掉了奈揍,然后在changeValue中讓buffer指向了其他對(duì)象,也不會(huì)影響到main方法中的buffer赋续,原因在于java采用的是值傳遞男翰,對(duì)于引用變量,傳遞的是引用的值纽乱,也就是說讓實(shí)參和形參同時(shí)指向了同一個(gè)對(duì)象蛾绎,因此讓形參重新指向另一個(gè)對(duì)象對(duì)實(shí)參并沒有任何影響。
轉(zhuǎn)載:淺析Java中的final關(guān)鍵字 - 海 子 - 博客園