重新認(rèn)識(shí)java(七) ---- final 關(guān)鍵字

你總以為你會(huì)了,其實(shí)你只是一知半解膏萧。

final 關(guān)鍵字概覽

final關(guān)鍵字可用于聲明屬性漓骚、方法蝌衔、參數(shù)和類,分別表示屬性不可變认境、方法不可覆蓋胚委、參數(shù)不可變和類不可以繼承。

我們來(lái)分別看看它的用法叉信。

final關(guān)鍵字是一個(gè)比較簡(jiǎn)單的知識(shí)點(diǎn)亩冬,所以這篇文章我寫的比較舒服,你看著也比較舒服硼身。因?yàn)楣杓保芎?jiǎn)單呀~

final 屬性

被final修飾的屬性不可變。這種不可變的屬性佳遂,我們可以稱之為“常量”营袜。這種常量大體上有兩種表現(xiàn)形式。先來(lái)看下面的代碼:

public class FinalAttribute {
    private final String attribute_a = "chengfan";
   
    public void test(){
        //attribute_a = "zhangbingxiao"; 不可以這樣寫
    }
}

這是最基本的final屬性丑罪,在定義的時(shí)候初始化荚板,并且在編譯期值就已經(jīng)確定,不能修改吩屹。

我們?cè)賮?lái)看一種:

public class FinalAttributeB {
    private final String attribute_b;
    
    public FinalAttributeB(String attribute_b){
        this.attribute_b = attribute_b;
    }
    
    public void test(){
        //attribute_b = "zhangbingxiao";
    }
    
    public void test(String attribute_b){
        //this.attribute_b = attribute_b;
    }
}

這種final屬性在編譯期間是無(wú)法確定屬性值的跪另,只有運(yùn)行的時(shí)候才可以確定(通過(guò)構(gòu)造器初始化屬性)。同樣煤搜,屬性一經(jīng)初始化后就不可以改變免绿,所以下面的test方法都無(wú)法修改final屬性。

上一篇文章中擦盾,我們講了代碼塊嘲驾,那么能不能使用代碼塊來(lái)初始化final屬性呢?答案當(dāng)然是可以的:

public class FinalAttributeB {
    private final String attribute_b;

    {
        attribute_b = "zhangbingxiao";
    }
    
    static {
        //attribute_b = "zhangbingxiao"; 
    }

//    public FinalAttributeB(String attribute_b){
//        this.attribute_b = attribute_b;
//    }
    
}

通過(guò)構(gòu)造代碼塊初始化final屬性也是可以的迹卢,但是這樣就不能再使用構(gòu)造函數(shù)初始化了辽故,因?yàn)闃?gòu)造代碼塊先于構(gòu)造函數(shù)執(zhí)行。而final屬性只能且必須初始化一次婶希。

你可能發(fā)現(xiàn)了榕暇,我寫了靜態(tài)代碼塊,但是注釋掉了喻杈。沒錯(cuò)彤枢,因?yàn)殪o態(tài)代碼塊只能初始化靜態(tài)屬性,我們?cè)谖恼伦詈笤儆懻撍?/p>

這種不在定義時(shí)初始化筒饰,而使用構(gòu)造函數(shù)初始化的缴啡,也稱為空白final變量。它為final在使用上提供了更大的靈活性瓷们,為此业栅,一個(gè)類中的final數(shù)據(jù)成員就可以實(shí)現(xiàn)依對(duì)象而有所不同秒咐,卻有保持其恒定不變的特征。

那除了構(gòu)造函數(shù)碘裕,有沒有別的方式也達(dá)到編譯時(shí)初始化呢携取?當(dāng)然有,比如你使用Random來(lái)初始化:

private final int attribute_c = new Random().nextInt();

這樣你只有在運(yùn)行的時(shí)候帮孔,才知道屬性值是多少雷滋。

剛剛我們研究的都是基本數(shù)據(jù)類型,那么文兢,引用數(shù)據(jù)類型呢晤斩?直接看代碼:

public class FinalAttributeC {
    private final Person person = new Person("zhangbingxiao");

    public void change(){
        person.setName("chengfan");
        System.out.println(person.getName());
    }
   //public void change(Person p){
   //this.person = p;
   //}

    public static void main(String[] args) {
        new FinalAttributeC().change();
    }
}
//結(jié)果 : chengfan

注釋掉的代碼是會(huì)報(bào)錯(cuò)的代碼,也就是說(shuō)引用類型person是不可以被修改的姆坚。從結(jié)果可以看出來(lái)澳泵,Person對(duì)象內(nèi)部的屬性被改變了。

所以兼呵,對(duì)于引用類型來(lái)說(shuō)兔辅,引用本身是不可以改變得,但是引用指向的對(duì)象是可以改變的击喂。

引用存在于棧中幢妄,而對(duì)象存在于堆中。引用的值是對(duì)象在堆中的地址茫负。在本質(zhì)上,final修飾的是引用乎赴,而不是對(duì)象忍法。所以引用的那個(gè)地址不可以變,而和對(duì)象沒多大關(guān)系榕吼。

舉個(gè)簡(jiǎn)單的例子饿序,一個(gè)人是一個(gè)對(duì)象,他會(huì)穿上衣羹蚣,褲子原探,鞋子,這些事人這個(gè)對(duì)象的屬性顽素。而人的名字是引用咽弦。當(dāng)你一生下來(lái),名字確定(引用確定)胁出,你可以隨便換衣服型型,但是你的名字還是那個(gè)。

我就舉個(gè)例子全蝶,別和我抬杠闹蒜。寺枉。什么可以去改名字,重名啥的绷落。姥闪。你理解了final引用類型這個(gè)知識(shí)就好了。

final 方法

當(dāng)一個(gè)方法聲明為final時(shí)砌烁,該方法不能被任何子類重寫筐喳,本類可以重載,但是子類可以使用這個(gè)方法往弓。

public class FinalMethod {
    public final void test(){

    }

    public void test(int i){

    }
}

class Test extends FinalMethod{

    //public void test(){} 不可以重寫

    @Override
    public void test(int i) {
        super.test(i);
    }
    public void test(int i,int j) {
        
    }
}

被final修飾的方法疏唾,不可以被重寫。但是不影響本類的重載以及重載函數(shù)的重寫函似。

這里有一種稱為內(nèi)聯(lián)(inline)的機(jī)制槐脏,當(dāng)調(diào)用一個(gè)被聲明為final的方法時(shí),直接將方法主體插入到調(diào)用處撇寞,而不是進(jìn)行正常的方法調(diào)用(類似于c++的內(nèi)聯(lián))顿天,這樣有利于提高程序的效率。

但是如果方法過(guò)于龐大蔑担,可能看不到內(nèi)聯(lián)調(diào)用帶來(lái)的任何性能提升牌废。在最近的Java版本中,不需要使用final方法進(jìn)行這些優(yōu)化了啤握。

final 參數(shù)

當(dāng)一個(gè)方法的形參被final修飾的時(shí)候鸟缕,這個(gè)參數(shù)在該方法內(nèi)不可以被修改。

public class FinalParam {
    public void test(final int a ){
        //a = 10; 值不可以被修改
    }
    public void test(final Person p){
        //p = new Person("zhangbingxiao"); 引用本身不可以被修改
        p.setName("zhangbingxiao");  //引用所指向的對(duì)象可以被修改
    }
}

對(duì)于引用數(shù)據(jù)類型的修改規(guī)則同final屬性一樣排抬。

final修飾參數(shù)在內(nèi)部類中是非常有用的懂从,在匿名內(nèi)部類中,為了保持參數(shù)的一致性蹲蒲,若所在的方法的形參需要被內(nèi)部類里面使用時(shí)番甩,該形參必須為final。

這個(gè)知識(shí)會(huì)在講解內(nèi)部類的時(shí)候進(jìn)行詳細(xì)的討論届搁,感興趣的可以先自行研究缘薛。

final修飾局部變量

final修飾局部變量時(shí)只能初始化(賦值)一次,可以不立即初始化卡睦。

public class StaticPartAttr {
    public void test(){
        final int a ;
        final int b = 2;
        
        a = 3;
        //a = 4;  報(bào)錯(cuò)  
        //b = 5;  報(bào)錯(cuò)
    }
}

被final修飾的局部變量宴胧,只能賦值一次。

你也可以一直不初始化表锻,但是不不賦值牺汤,定義這個(gè)變量還有什么用呢?

final 類

被final修飾的類不可以被繼承浩嫌,所有方法不能被重寫(廢話檐迟,都不能繼承了补胚,哪來(lái)的重寫)。但是這并不表示類內(nèi)部的屬性也是不可修改的追迟,除非這個(gè)屬性也被final修飾溶其。這點(diǎn)在jdk里有很多應(yīng)用,比如我們熟知的String敦间,Integer等類都被final修飾瓶逃。

final類有很多好處,譬如它們的對(duì)象是只讀的廓块,可以在多線程環(huán)境下安全的共享厢绝,不用額外的同步開銷等等。

如何寫一個(gè)不可變類呢带猴?

  • 將類聲明為final昔汉,所以它不能被繼承
  • 將所有的成員聲明為私有的,這樣就不允許直接訪問(wèn)這些成員
  • 對(duì)變量不要提供setter方法
  • 將所有可變的成員聲明為final拴清,這樣只能對(duì)它們賦值一次
  • 通過(guò)構(gòu)造器初始化所有成員靶病,進(jìn)行深拷貝(deep copy)
  • 在getter方法中,不要直接返回對(duì)象本身口予,而是克隆對(duì)象娄周,并返回對(duì)象的拷貝

詳情--->丟個(gè)鏈接趕緊跑

值得注意的是沪停,一個(gè)類不可以既被abstract修飾又被final修飾煤辨。因?yàn)閒inal類不可以被繼承,而abstract類需要被繼承木张。關(guān)于抽象類掷酗,我們會(huì)在下篇文章中詳細(xì)講解。

**final 與 static **

當(dāng)final和static同時(shí)使用的時(shí)候窟哺,我們所熟知的“全局常量”就出現(xiàn)了:一個(gè)可以到處使用并且不可以改變的屬性,比如我們熟知的Math.PI技肩,Math.E且轨。

上面我們說(shuō)到了靜態(tài)代碼塊初始化final變量的問(wèn)題。

public class FinalStatic {
    private final static double PI = 3.14;
    private final static double E;
    private final static double C ; //這里會(huì)報(bào)錯(cuò)
    
    static {
        E = 2.71;
    }
    
    public FinalStatic(double c){
        C = c;
        //PI = C;   這里會(huì)報(bào)錯(cuò)
    }
}

對(duì)于靜態(tài)final變量虚婿,我們可以直接初始化旋奢,或者使用靜態(tài)代碼塊。而不可以使用構(gòu)造函數(shù)或者構(gòu)造代碼塊然痊。

因?yàn)閟tatic要求在編譯期間就確定值至朗,然后放入靜態(tài)區(qū)。而構(gòu)造函數(shù)和構(gòu)造代碼塊發(fā)生在運(yùn)行期間剧浸。所以不存在空白靜態(tài)final锹引。

final和private

類中所有的private方法都隱式的指定為final的矗钟,由于無(wú)法取用private方法,所以也就無(wú)法覆蓋它嫌变,可以對(duì)private方法添加final修飾符吨艇,但并沒有添加任何額外意義。

總結(jié)

關(guān)于final的重要知識(shí)點(diǎn)

  • final關(guān)鍵字可以用于成員變量腾啥、本地變量东涡、方法以及類。
  • final成員變量必須在聲明的時(shí)候初始化或者在構(gòu)造器中初始化倘待,否則就會(huì)報(bào)編譯錯(cuò)誤疮跑。
  • 你不能夠?qū)inal變量再次賦值。
  • 本地變量必須在聲明時(shí)賦值凸舵。
  • 在匿名類中所有變量都必須是final變量祖娘。
  • final方法不能被重寫。
  • final類不能被繼承贞间。
  • 接口中聲明的所有變量本身是final的贿条。
  • final和abstract這兩個(gè)關(guān)鍵字是反相關(guān)的,final類就不可能是abstract的增热。
  • final方法在編譯階段綁定整以,稱為靜態(tài)綁定(static binding)。
  • 沒有在聲明時(shí)初始化final變量的稱為空白final變量(blank final variable)峻仇,它們必須在構(gòu)造器中或者代碼塊中初始化公黑。
  • 將類、方法摄咆、變量聲明為final能夠提高性能凡蚜,這樣JVM就有機(jī)會(huì)進(jìn)行估計(jì),然后優(yōu)化吭从。
  • 按照J(rèn)ava代碼慣例朝蜘,final變量就是常量,而且通常常量名要大寫涩金。

本文內(nèi)容到此結(jié)束谱醇。如果文章有錯(cuò)誤或者你有更好的理解方式,請(qǐng)及時(shí)與我聯(lián)系~歡迎指出錯(cuò)誤步做,比較我也是個(gè)學(xué)習(xí)的人而不是大神副渴。

轉(zhuǎn)載請(qǐng)注明出處
http://www.reibang.com/p/e43c3273d7d5

<strong>看完了,點(diǎn)個(gè)贊唄~</strong>

最后編輯于
?著作權(quán)歸作者所有,轉(zhuǎn)載或內(nèi)容合作請(qǐng)聯(lián)系作者
  • 序言:七十年代末全度,一起剝皮案震驚了整個(gè)濱河市煮剧,隨后出現(xiàn)的幾起案子,更是在濱河造成了極大的恐慌,老刑警劉巖勉盅,帶你破解...
    沈念sama閱讀 206,214評(píng)論 6 481
  • 序言:濱河連續(xù)發(fā)生了三起死亡事件佑颇,死亡現(xiàn)場(chǎng)離奇詭異,居然都是意外死亡菇篡,警方通過(guò)查閱死者的電腦和手機(jī)漩符,發(fā)現(xiàn)死者居然都...
    沈念sama閱讀 88,307評(píng)論 2 382
  • 文/潘曉璐 我一進(jìn)店門,熙熙樓的掌柜王于貴愁眉苦臉地迎上來(lái)驱还,“玉大人嗜暴,你說(shuō)我怎么就攤上這事∫轶。” “怎么了闷沥?”我有些...
    開封第一講書人閱讀 152,543評(píng)論 0 341
  • 文/不壞的土叔 我叫張陵,是天一觀的道長(zhǎng)咐容。 經(jīng)常有香客問(wèn)我舆逃,道長(zhǎng),這世上最難降的妖魔是什么戳粒? 我笑而不...
    開封第一講書人閱讀 55,221評(píng)論 1 279
  • 正文 為了忘掉前任路狮,我火速辦了婚禮,結(jié)果婚禮上蔚约,老公的妹妹穿的比我還像新娘奄妨。我一直安慰自己,他們只是感情好苹祟,可當(dāng)我...
    茶點(diǎn)故事閱讀 64,224評(píng)論 5 371
  • 文/花漫 我一把揭開白布砸抛。 她就那樣靜靜地躺著,像睡著了一般树枫。 火紅的嫁衣襯著肌膚如雪直焙。 梳的紋絲不亂的頭發(fā)上,一...
    開封第一講書人閱讀 49,007評(píng)論 1 284
  • 那天砂轻,我揣著相機(jī)與錄音奔誓,去河邊找鬼。 笑死搔涝,一個(gè)胖子當(dāng)著我的面吹牛厨喂,可吹牛的內(nèi)容都是我干的。 我是一名探鬼主播体谒,決...
    沈念sama閱讀 38,313評(píng)論 3 399
  • 文/蒼蘭香墨 我猛地睜開眼,長(zhǎng)吁一口氣:“原來(lái)是場(chǎng)噩夢(mèng)啊……” “哼臼婆!你這毒婦竟也來(lái)了抒痒?” 一聲冷哼從身側(cè)響起,我...
    開封第一講書人閱讀 36,956評(píng)論 0 259
  • 序言:老撾萬(wàn)榮一對(duì)情侶失蹤颁褂,失蹤者是張志新(化名)和其女友劉穎故响,沒想到半個(gè)月后傀广,有當(dāng)?shù)厝嗽跇淞掷锇l(fā)現(xiàn)了一具尸體,經(jīng)...
    沈念sama閱讀 43,441評(píng)論 1 300
  • 正文 獨(dú)居荒郊野嶺守林人離奇死亡彩届,尸身上長(zhǎng)有42處帶血的膿包…… 初始之章·張勛 以下內(nèi)容為張勛視角 年9月15日...
    茶點(diǎn)故事閱讀 35,925評(píng)論 2 323
  • 正文 我和宋清朗相戀三年伪冰,在試婚紗的時(shí)候發(fā)現(xiàn)自己被綠了。 大學(xué)時(shí)的朋友給我發(fā)了我未婚夫和他白月光在一起吃飯的照片樟蠕。...
    茶點(diǎn)故事閱讀 38,018評(píng)論 1 333
  • 序言:一個(gè)原本活蹦亂跳的男人離奇死亡贮聂,死狀恐怖,靈堂內(nèi)的尸體忽然破棺而出寨辩,到底是詐尸還是另有隱情吓懈,我是刑警寧澤,帶...
    沈念sama閱讀 33,685評(píng)論 4 322
  • 正文 年R本政府宣布靡狞,位于F島的核電站耻警,受9級(jí)特大地震影響,放射性物質(zhì)發(fā)生泄漏甸怕。R本人自食惡果不足惜甘穿,卻給世界環(huán)境...
    茶點(diǎn)故事閱讀 39,234評(píng)論 3 307
  • 文/蒙蒙 一、第九天 我趴在偏房一處隱蔽的房頂上張望梢杭。 院中可真熱鬧温兼,春花似錦、人聲如沸式曲。這莊子的主人今日做“春日...
    開封第一講書人閱讀 30,240評(píng)論 0 19
  • 文/蒼蘭香墨 我抬頭看了看天上的太陽(yáng)吝羞。三九已至兰伤,卻和暖如春,著一層夾襖步出監(jiān)牢的瞬間钧排,已是汗流浹背敦腔。 一陣腳步聲響...
    開封第一講書人閱讀 31,464評(píng)論 1 261
  • 我被黑心中介騙來(lái)泰國(guó)打工, 沒想到剛下飛機(jī)就差點(diǎn)兒被人妖公主榨干…… 1. 我叫王不留恨溜,地道東北人符衔。 一個(gè)月前我還...
    沈念sama閱讀 45,467評(píng)論 2 352
  • 正文 我出身青樓,卻偏偏與公主長(zhǎng)得像糟袁,于是被迫代替她去往敵國(guó)和親判族。 傳聞我的和親對(duì)象是個(gè)殘疾皇子,可洞房花燭夜當(dāng)晚...
    茶點(diǎn)故事閱讀 42,762評(píng)論 2 345

推薦閱讀更多精彩內(nèi)容

  • 1. Java基礎(chǔ)部分 基礎(chǔ)部分的順序:基本語(yǔ)法项戴,類相關(guān)的語(yǔ)法形帮,內(nèi)部類的語(yǔ)法,繼承相關(guān)的語(yǔ)法,異常的語(yǔ)法辩撑,線程的語(yǔ)...
    子非魚_t_閱讀 31,581評(píng)論 18 399
  • 一:java概述:1界斜,JDK:Java Development Kit,java的開發(fā)和運(yùn)行環(huán)境合冀,java的開發(fā)工...
    ZaneInTheSun閱讀 2,629評(píng)論 0 11
  • Advanced Language Features 知識(shí)點(diǎn):一. static修飾符 static修飾符可以用來(lái)...
    風(fēng)景涼閱讀 434評(píng)論 0 0
  • 123.繼承 一個(gè)類可以從另外一個(gè)類繼承方法,屬性和其他特征各薇。當(dāng)一個(gè)類繼承另外一個(gè)類時(shí), 繼承類叫子類, 被繼承的...
    無(wú)灃閱讀 1,380評(píng)論 2 4
  • 13年的學(xué)習(xí)生涯峭判,就這樣過(guò)去,其間的苦與樂(lè)晰洒、淚與笑朝抖,你我最清楚。 01 6歲那年谍珊,我才上幼稚園大班治宣,卻已懂得許多人...
    夏一憶閱讀 442評(píng)論 0 3