在程序設(shè)計(jì)中,我們有時(shí)可能希望某些數(shù)據(jù)是不能夠改變的,這個(gè)時(shí)候 final 就有用武之地了。final 是 Java 的關(guān)鍵字,它所表示的是“這部分是無(wú)法修改的”。不想被改變的原因有兩個(gè):效率、設(shè)計(jì)。使用到 final 的有三種情況:數(shù)據(jù)、方法、類(lèi)。
有時(shí)候數(shù)據(jù)的恒定不變是很有用的,它能夠減輕系統(tǒng)運(yùn)行時(shí)的負(fù)擔(dān)。對(duì)于這些恒定不變的數(shù)據(jù)我可以叫做“常量”?!俺A俊敝饕獞?yīng)用與以下兩個(gè)地方:
1、編譯期常量,永遠(yuǎn)不可改變。
2、運(yùn)行期初始化時(shí),我們希望它不會(huì)被改變。
對(duì)于編譯期常量,它在類(lèi)加載的過(guò)程就已經(jīng)完成了初始化,所以當(dāng)類(lèi)加載完成后是不可更改的,編譯期可以將它代入到任何用到它的計(jì)算式中,也就是說(shuō)可以在編譯期執(zhí)行計(jì)算式。當(dāng)然對(duì)于編譯期常量,只能使用基本類(lèi)型,而且必須要在定義時(shí)進(jìn)行初始化。
有些變量,我們希望它可以根據(jù)對(duì)象的不同而表現(xiàn)不同,但同時(shí)又不希望它被改變,這個(gè)時(shí)候我們就可以使用運(yùn)行期常量。對(duì)于運(yùn)行期常量,它既可是基本數(shù)據(jù)類(lèi)型,也可是引用數(shù)據(jù)類(lèi)型?;緮?shù)據(jù)類(lèi)型不可變的是其內(nèi)容,而引用數(shù)據(jù)類(lèi)型不可變的是其引用,引用所指定的對(duì)象內(nèi)容是可變的。
public class Person {
private String name;
Person(String name){
this.name = name;
}
public String getName() {
return name;
}
public void setName(String name) {
this.name = name;
}
}
public class FinalTest {
private final String final_01 = "chenssy"; //編 譯期常量,必須要進(jìn)行初始化,且不可更改
private final String final_02; //構(gòu)造器常量,在實(shí)例化一個(gè)對(duì)象時(shí)被初始化
private static Random random = new Random();
private final int final_03 = random.nextInt(50); //使用隨機(jī)數(shù)來(lái)進(jìn)行初始化
//引用
public final Person final_04 = new Person("chen_ssy"); //final指向引用數(shù)據(jù)類(lèi)型
FinalTest(String final_02){
this.final_02 = final_02;
}
public String toString(){
return "final_01 = " + final_01 +" final_02 = " + final_02 + " final_03 = " + final_03 +
" final_04 = " + final_04.getName();
}
public static void main(String[] args) {
System.out.println("------------第一次創(chuàng)建對(duì)象------------");
FinalTest final1 = new FinalTest("cm");
System.out.println(final1);
System.out.println("------------第二次創(chuàng)建對(duì) 象------------");
FinalTest final2 = new FinalTest("zj");
System.out.println(final2);
System.out.println("------------修改引用對(duì) 象--------------");
final2.final_04.setName("chenssy");
System.out.println(final2);
}
}
------------------
Output:
------------第一次創(chuàng)建對(duì)象------------
final_01 = chenssy final_02 = cm final_03 = 34 final_04 = chen_ssy
------------第二次創(chuàng)建對(duì)象------------
final_01 = chenssy final_02 = zj final_03 = 46 final_04 = chen_ssy
------------修改引用對(duì)象--------------
final_01 = chenssy final_02 = zj final_03 = 46 final_04 = chenssy
這里只闡述一點(diǎn)就是:不要以為某些數(shù)據(jù)是 final 就可以在編譯期知道其值,通過(guò) final_03 我們就知道了,在這里是使用隨機(jī)數(shù)其進(jìn)行初始化,他要在運(yùn)行期才能知道其值。
所有被 final 標(biāo)注的方法都是不能被繼承、更改的,所以對(duì)于 final 方法使用的第一個(gè)原因就是方法鎖定,以防止任何子類(lèi)來(lái)對(duì)它的修改。至于第二個(gè)原因就是效率問(wèn)題,鄙人對(duì)這個(gè)效率問(wèn)題理解的不是很清楚,在網(wǎng)上摘抄這段話(huà):
在 Java 的早期實(shí)現(xiàn)中,如果將一個(gè)方法指明為 final,就是同意編譯器將針對(duì)該方法的所有調(diào)用都轉(zhuǎn)為內(nèi)嵌調(diào)用。當(dāng)編譯器發(fā)現(xiàn)一個(gè) final 方法調(diào)用命令時(shí),它會(huì)根據(jù)自己的謹(jǐn)慎判斷,跳過(guò)插入程序代碼這種正常的調(diào)用方式而執(zhí)行方法調(diào)用機(jī)制(將參數(shù)壓入棧,跳至方法代碼處執(zhí)行,然后跳回并清理?xiàng)V械膮?shù),處理返回值),并且以方法體中的實(shí)際代碼的副本來(lái)代替方法調(diào)用。這將消除方法調(diào)用的開(kāi)銷(xiāo)。當(dāng)然,如果一個(gè)方法很大,你的程序代碼會(huì)膨脹,因而可能看不到內(nèi)嵌所帶來(lái)的性能上的提高,因?yàn)樗鶐?lái)的性能會(huì)花費(fèi)于方法內(nèi)的時(shí)間量而被縮減。
對(duì)這段話(huà)理解我不是很懂就照搬了,那位 Java 牛人可以解釋解釋下!!
父類(lèi)的 final 方法是不能被子類(lèi)所覆蓋的,也就是說(shuō)子類(lèi)是不能夠存在和父類(lèi)一模一樣的方法的。
public class Custom extends Person{
public void method1(){
System.out.println("Person's method1....");
}
// Cannot override the final method from person:子類(lèi)不能覆蓋父類(lèi)的final方法
// public void method2(){
// System.out.println("Person's method2...");
// }
}
如果某個(gè)類(lèi)用 final 修改,表明該類(lèi)是最終類(lèi),它不希望也不允許其他來(lái)繼承它。在程序設(shè)計(jì)中處于安全或者其他原因,我們不允許該類(lèi)存在任何變化,也不希望它有子類(lèi),這個(gè)時(shí)候就可以使用 final 來(lái)修飾該類(lèi)了。
對(duì)于 final 修飾的類(lèi)來(lái)說(shuō),它的成員變量可以為 final,也可以為非 final。如果定義為 final,那么 final 數(shù)據(jù)的規(guī)則同樣適合它。而它的方法則會(huì)自動(dòng)的加上 final,因?yàn)?final 類(lèi)是無(wú)法被繼承,所以這個(gè)是默認(rèn)的。
在實(shí)際應(yīng)用中,我們除了可以用 final 修飾成員變量、成員方法、類(lèi),還可以修飾參數(shù)、若某個(gè)參數(shù)被 final 修飾了,則代表了該參數(shù)是不可改變的。
如果在方法中我們修改了該參數(shù),則編譯器會(huì)提示你:The final local variable i cannot be assigned. It must be blank and not using a compound assignment。
public class Custom {
public void test(final int i){
//i++; ---final參數(shù)不可改變
System.out.println(i);
}
public void test(final Person p){
//p = new Person(); --final參數(shù)不可變
p.setName("chenssy");
}
}
同 final 修飾參數(shù)在內(nèi)部類(lèi)中是非常有用的,在匿名內(nèi)部類(lèi)中,為了保持參數(shù)的一致性,若所在的方法的形參需要被內(nèi)部類(lèi)里面使用時(shí),該形參必須為 final。詳情參看:http://www.cnblogs.com/chenssy/p/3390871.html 。
final 和 static 在一起使用就會(huì)發(fā)生神奇的化學(xué)反應(yīng),他們同時(shí)使用時(shí)即可修飾成員變量,也可修飾成員方法。
對(duì)于成員變量,該變量一旦賦值就不能改變,我們稱(chēng)它為“全局常量”??梢酝ㄟ^(guò)類(lèi)名直接訪(fǎng)問(wèn)。
對(duì)于成員方法,則是不可繼承和改變??梢酝ㄟ^(guò)類(lèi)名直接訪(fǎng)問(wèn)。
更多:
Java 提高篇——關(guān)鍵字static:http://www.cnblogs.com/chenssy/p/3386721.html
鞏固基礎(chǔ),提高技術(shù),不懼困難,攀登高峰?。。。。?!