Java不可變類

0. 幾個問題

  1. 什么是不可變類?
  2. 不可變類的優缺點是什么?
  3. 常見的不可變類有哪些?String為什么要設計成不可變類?
  4. 如何自己設計一個不可變類?
    帶著這幾個問題閱讀本文以期能對Java的不可變類有一個全面的了解。

1. 什么是不可變類

不可變類是指類的實例一旦創建后,不能改變其成員變量的值
與之對應的,可變類的實例創建后可以改變其成員變量的值。

2. 不可變類的優缺點

  • 優點:
    1. 效率,例如字符串常量池,字符串常量池可以將一些字符常量放在常量池中重復使用,避免每次都重新創建相同的對象、節省存儲空間。但如果字符串是可變的,此時相同內容的String還指向常量池的同一個內存空間,當某個變量改變了該內存的值時,其他遍歷的值也會發生改變。所以不符合常量池設計的初衷。
    2. 安全性,不可變對象天生是線程安全的,在不同線程共享對象,不需要同步機制,因為對象的值是固定的。
  • 缺點:
    1. 資源開銷,對象需要頻繁的修改屬性,則每一次修改都會新創建一個對象,產生大量的資源開銷。

3. 常見的不可變類有哪些?String為什么要設計成不可變類?

常見的不可變類:String Integer Long等類型
String設計成不可變類主要是出于效率和安全性考慮。

4. 如何自己設計一個不可變類?

  1. 類使用final修飾符修飾,保證類不能被繼承。
    如果類可以被繼承會破壞類的不可變性機制,只要繼承類覆蓋父類的方法并且繼承類可以改變成員變量值,那么一旦子類以父類的形式出現時,不能保證當前類是否可變。
  2. 類的成員變量都應該是private final的,保證成員變量不可改變。
  3. 任何獲取/修改屬性的方法都不應作用于屬性本身。
    • 不提供修改成員變量的方法,例如setter方法。
    • getter方法不能返回對象本身,要返回對象的拷貝,防止對象外泄。
    • 修改對象的屬性時要返回新對象
  4. 對成員變量的初始化通過構造器進行,并進行深拷貝。
    如果使用傳入的參數直接賦值,則傳遞的只是引用,仍然可以通過外部變量改變它的值。

5. String類的不可變性分析

public final class String
    implements java.io.Serializable, Comparable<String>, CharSequence {
    /** The value is used for character storage. */
    private final char value[];
    ...
    /**
     * Allocates a new {@code String} so that it represents the sequence of
     * characters currently contained in the character array argument. The
     * contents of the character array are copied; subsequent modification of
     * the character array does not affect the newly created string.
     *
     * @param  value
     *         The initial value of the string
     */
    public String(char value[]) {
        this.value = Arrays.copyOf(value, value.length);
    }
    ...
    /**
     * Converts this string to a new character array.
     *
     * @return  a newly allocated character array whose length is the length
     *          of this string and whose contents are initialized to contain
     *          the character sequence represented by this string.
     */
    public char[] toCharArray() {
        // Cannot use Arrays.copyOf because of class initialization order issues
        char result[] = new char[value.length];
        System.arraycopy(value, 0, result, 0, value.length);
        return result;
    }
}

可以看出String類的設計遵循以下原則:

  1. String類被final修飾,不可繼承
  2. string內部所有成員都設置為私有變量
  3. 不存在value的setter
  4. 并將value和offset設置為final。
  5. 當傳入可變數組value[]時,進行copy而不是直接將value[]復制給內部變量.
  6. 獲取value時不是直接返回對象引用,而是返回對象的copy.

這都符合上面總結的不變類型的特性,也保證了String類型是不可變的類。

但是,即使是不可變類,通過反射仍然可以改變其屬性的值

String s = "hello world";
System.out.println("s=" + s);
try {
    Field v = String.class.getDeclaredField("value");
    v.setAccessible(true);
    char[] c = (char[]) v.get(s);
    c[5] = '_';
    System.out.println("s=" + s);
} catch (NoSuchFieldException nfe) {
    nfe.printStackTrace();
} catch (IllegalAccessException iae) {
    iae.printStackTrace();
}

輸出結果為:

s=hello world
s=hello_world

6. 參考

  1. String源碼
  2. JAVA不可變類(immutable)機制與String的不可變性
  3. 為什么Java的string類要設成immutable(不可變的)
  4. JAVA的可變類與不可變類
  5. Java不可變類機制淺析
最后編輯于
?著作權歸作者所有,轉載或內容合作請聯系作者
平臺聲明:文章內容(如有圖片或視頻亦包括在內)由作者上傳并發布,文章內容僅代表作者本人觀點,簡書系信息發布平臺,僅提供信息存儲服務。
  • 序言:七十年代末,一起剝皮案震驚了整個濱河市,隨后出現的幾起案子,更是在濱河造成了極大的恐慌,老刑警劉巖,帶你破解...
    沈念sama閱讀 230,622評論 6 544
  • 序言:濱河連續發生了三起死亡事件,死亡現場離奇詭異,居然都是意外死亡,警方通過查閱死者的電腦和手機,發現死者居然都...
    沈念sama閱讀 99,716評論 3 429
  • 文/潘曉璐 我一進店門,熙熙樓的掌柜王于貴愁眉苦臉地迎上來,“玉大人,你說我怎么就攤上這事。” “怎么了?”我有些...
    開封第一講書人閱讀 178,746評論 0 383
  • 文/不壞的土叔 我叫張陵,是天一觀的道長。 經常有香客問我,道長,這世上最難降的妖魔是什么? 我笑而不...
    開封第一講書人閱讀 63,991評論 1 318
  • 正文 為了忘掉前任,我火速辦了婚禮,結果婚禮上,老公的妹妹穿的比我還像新娘。我一直安慰自己,他們只是感情好,可當我...
    茶點故事閱讀 72,706評論 6 413
  • 文/花漫 我一把揭開白布。 她就那樣靜靜地躺著,像睡著了一般。 火紅的嫁衣襯著肌膚如雪。 梳的紋絲不亂的頭發上,一...
    開封第一講書人閱讀 56,036評論 1 329
  • 那天,我揣著相機與錄音,去河邊找鬼。 笑死,一個胖子當著我的面吹牛,可吹牛的內容都是我干的。 我是一名探鬼主播,決...
    沈念sama閱讀 44,029評論 3 450
  • 文/蒼蘭香墨 我猛地睜開眼,長吁一口氣:“原來是場噩夢啊……” “哼!你這毒婦竟也來了?” 一聲冷哼從身側響起,我...
    開封第一講書人閱讀 43,203評論 0 290
  • 序言:老撾萬榮一對情侶失蹤,失蹤者是張志新(化名)和其女友劉穎,沒想到半個月后,有當地人在樹林里發現了一具尸體,經...
    沈念sama閱讀 49,725評論 1 336
  • 正文 獨居荒郊野嶺守林人離奇死亡,尸身上長有42處帶血的膿包…… 初始之章·張勛 以下內容為張勛視角 年9月15日...
    茶點故事閱讀 41,451評論 3 361
  • 正文 我和宋清朗相戀三年,在試婚紗的時候發現自己被綠了。 大學時的朋友給我發了我未婚夫和他白月光在一起吃飯的照片。...
    茶點故事閱讀 43,677評論 1 374
  • 序言:一個原本活蹦亂跳的男人離奇死亡,死狀恐怖,靈堂內的尸體忽然破棺而出,到底是詐尸還是另有隱情,我是刑警寧澤,帶...
    沈念sama閱讀 39,161評論 5 365
  • 正文 年R本政府宣布,位于F島的核電站,受9級特大地震影響,放射性物質發生泄漏。R本人自食惡果不足惜,卻給世界環境...
    茶點故事閱讀 44,857評論 3 351
  • 文/蒙蒙 一、第九天 我趴在偏房一處隱蔽的房頂上張望。 院中可真熱鬧,春花似錦、人聲如沸。這莊子的主人今日做“春日...
    開封第一講書人閱讀 35,266評論 0 28
  • 文/蒼蘭香墨 我抬頭看了看天上的太陽。三九已至,卻和暖如春,著一層夾襖步出監牢的瞬間,已是汗流浹背。 一陣腳步聲響...
    開封第一講書人閱讀 36,606評論 1 295
  • 我被黑心中介騙來泰國打工, 沒想到剛下飛機就差點兒被人妖公主榨干…… 1. 我叫王不留,地道東北人。 一個月前我還...
    沈念sama閱讀 52,407評論 3 400
  • 正文 我出身青樓,卻偏偏與公主長得像,于是被迫代替她去往敵國和親。 傳聞我的和親對象是個殘疾皇子,可洞房花燭夜當晚...
    茶點故事閱讀 48,643評論 2 380

推薦閱讀更多精彩內容

  • 1. Java基礎部分 基礎部分的順序:基本語法,類相關的語法,內部類的語法,繼承相關的語法,異常的語法,線程的語...
    子非魚_t_閱讀 31,738評論 18 399
  • 一、Java 簡介 Java是由Sun Microsystems公司于1995年5月推出的Java面向對象程序設計...
    子非魚_t_閱讀 4,240評論 1 44
  • 注:都是在百度搜索整理的答案,如有侵權和錯誤,希告知更改。 一、哪些情況下的對象會被垃圾回收機制處理掉 ?當對象對...
    Jenchar閱讀 3,239評論 3 2
  • 一個人的精神發育史,應該是一個人的閱讀史。一個民族的精神境界,在很大程度上取決于全民族的閱讀水平。 面對這個提問,...
    瑩光灼華閱讀 4,245評論 191 219
  • 大雁塔。。唐城墻遺址公園。。曲江池遺址公園。。永興坊肉夾饃 俺們本來今天想要去陜博的,但是陜博人實在是太多了,沒招...
    不變的是那對wx閱讀 274評論 0 1