淺談為什么Java里面String類是不可變的
在Java里面String類型是不可變對(duì)象,這一點(diǎn)毫無(wú)疑問(wèn),那么為什么Java語(yǔ)言的設(shè)計(jì)者要把String類型設(shè)計(jì)成不可變對(duì)象呢?這是一個(gè)值得思考的問(wèn)題
Java語(yǔ)言的創(chuàng)建者James Gosling,曾經(jīng)在一次采訪中被人問(wèn)到:什么時(shí)候應(yīng)該使用不可變對(duì)象(immutable object),他回答:任何可以使用的時(shí)候都會(huì)使用。
在這之前,我們先來(lái)簡(jiǎn)單了解一下,什么是不可變對(duì)象?
不可變對(duì)象指的是在對(duì)象創(chuàng)建之后,對(duì)象的內(nèi)部狀態(tài)以及對(duì)象的內(nèi)存指針地址都不不能被改變。在Java里面final關(guān)鍵字就是用來(lái)輔助創(chuàng)建不可變對(duì)象的,但需要注意的是,對(duì)于基本類型被final修飾后,就徹底變成了不可變對(duì)象,而引用類型被final修飾后,僅僅是指針的內(nèi)存地址不能改變,如果想要變成徹底的不可變類型,要把該對(duì)象里面所有的字段都得用final聲明,包括嵌套的對(duì)象,否則對(duì)象的內(nèi)部狀態(tài)也是會(huì)變化的,這一點(diǎn)需要理解。
ok,下面我們來(lái)分析下為什么String是不可變的?
通過(guò)String源碼可以看到,String類型的底層是由final修飾的char數(shù)組存儲(chǔ)。
public final class String implements java.io.Serializable, Comparable<String>, CharSequence { /** The value is used for character storage. */ private final char value[]; ........ }
String能被設(shè)計(jì)成不可變類型的一個(gè)重要前是因?yàn)樗蔷幊陶Z(yǔ)言里面使用頻率最高的一種類型。不可變類型帶來(lái)的好處,體現(xiàn)在四個(gè)方面,分別是:緩存,安全,同步和性能。
(一)緩存
在JVM的運(yùn)行時(shí)數(shù)據(jù)區(qū)域里面,有一個(gè)專門的字符串常量池用來(lái)存儲(chǔ)字符串字面量,如下面一段代碼:
String s1 = "Hello World"; String s2 = "Hello World"; assertThat(s1 == s2).isTrue();
s1和s2變量指針的內(nèi)存地址其實(shí)是一樣的,也就是說(shuō)他們代表是同一個(gè)對(duì)象,這是jvm常量池做的優(yōu)化,當(dāng)?shù)谝粋€(gè)字面量聲明的時(shí)候,它的值會(huì)被字符串常量池存儲(chǔ),當(dāng)s2變量聲明的時(shí)候,jvm發(fā)現(xiàn)常量池已經(jīng)存在該對(duì)象,所以就不會(huì)再創(chuàng)建一次,而是直接將一樣的內(nèi)存指針賦值給s2變量,從避免了重復(fù)創(chuàng)建對(duì)象,節(jié)省了內(nèi)存空間。
此外,由于字符串的不可變性,從而可以讓其hashCode也被緩存,在Java里面哈希類數(shù)據(jù)結(jié)構(gòu)如HashMap, HashTable, HashSet其key用的最多的基本都是String類型,如此一來(lái)key的hashCode的也可以在第一次調(diào)用之后被緩存,之后直接使用無(wú)須重新生成,從而間接的提升訪問(wèn)效率。
(二)安全
不可變特性也能夠減少了應(yīng)用程序在運(yùn)行時(shí)間的安全問(wèn)題,如下面的一段代碼:
void criticalMethod(String userName) { // check if (!check(userName)) { throw new SecurityException(); } // query query(userName); }
在上面的一段代碼,在調(diào)用這個(gè)方法之后,先檢查用戶名,如果合法才可以繼續(xù)查詢相關(guān)數(shù)據(jù),如果String可變,那么攻擊者就可以在通過(guò)check驗(yàn)證之后,再改變查詢的用戶名,那么就會(huì)存在安全風(fēng)險(xiǎn),而不可變性能夠避免和減少這一情況。另一方面,如果String是可變的,那么同時(shí)運(yùn)行的其他線程如果修改這個(gè)值,就有可能導(dǎo)致混亂。
(三)同步
由于String類型的不可變性,使得String對(duì)象可以安全的在多個(gè)線程之間傳遞和訪問(wèn),也就是說(shuō)你在多線程中是不能改變字符串本身的值,而是在堆里面新創(chuàng)建一個(gè)字符串然后操作。當(dāng)然如果沒(méi)有final修飾,你是可以改變這個(gè)變量的引用地址,也就是說(shuō)你可以把新生成的內(nèi)存引用覆蓋原來(lái)的變量引用,但這里僅僅是引用,并不是變量的值。這一點(diǎn)要注意。
(四)性能
性能方面,其實(shí)前面已經(jīng)提到了,比如字符串的常量池節(jié)省內(nèi)存,緩存Hash類以字符串做key數(shù)據(jù)結(jié)構(gòu)的hashCode,從而提高訪問(wèn)性能等。由于字符串是編程語(yǔ)言里面最廣泛使用的數(shù)據(jù)結(jié)構(gòu),所以針對(duì)字符串的不可變性帶來(lái)的優(yōu)勢(shì),可以放大到整個(gè)運(yùn)行的應(yīng)用程序,從而帶來(lái)應(yīng)用程序整體的性能提升。
總結(jié):
本文主要介紹了Java語(yǔ)言里面String類型為什么設(shè)計(jì)成不可變類型,以及分析了不可變類型的帶來(lái)的主要優(yōu)勢(shì),需要注意的是雖然不可變類型能夠帶來(lái)不少的好處,但并不是說(shuō)其沒(méi)有弊端,不可變類型的每一次修改都需要在內(nèi)存中新生成一個(gè)對(duì)象,從另一個(gè)方面說(shuō)針對(duì)經(jīng)常變化的對(duì)象是不適合使用不可變類型的,這也是為什么Java里面還提供了可修改值的StringBuilder和StringBuffer類,這在實(shí)際開發(fā)中常常是需要根據(jù)具體情況權(quán)衡的。
以上所述是小編給大家介紹的為什么Java里面String類是不可變的詳解整合,希望對(duì)大家有所幫助,如果大家有任何疑問(wèn)請(qǐng)給我留言,小編會(huì)及時(shí)回復(fù)大家的。在此也非常感謝大家對(duì)腳本之家網(wǎng)站的支持!
相關(guān)文章
java 集合并發(fā)操作出現(xiàn)的異常ConcurrentModificationException
Map在遍歷時(shí)候通常 現(xiàn)獲得其鍵值的集合Set,然后用迭代器Iterator來(lái)對(duì)Map進(jìn)行遍歷。2009-06-06Java實(shí)現(xiàn)簡(jiǎn)單LRU緩存機(jī)制的方法
這篇文章主要介紹了Java實(shí)現(xiàn)簡(jiǎn)單LRU緩存機(jī)制的方法,文中通過(guò)示例代碼介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友們下面隨著小編來(lái)一起學(xué)習(xí)學(xué)習(xí)吧2020-05-05Java中常用解析工具jackson及fastjson的使用
今天給大家?guī)?lái)的是關(guān)于Java解析工具的相關(guān)知識(shí),文章圍繞著jackson及fastjson的使用展開,文中有非常詳細(xì)的介紹及代碼示例,需要的朋友可以參考下2021-06-06關(guān)于MyBatis plus條件構(gòu)造器的逐條詳解
什么是條件構(gòu)造器呢?簡(jiǎn)單來(lái)說(shuō),條件構(gòu)造器就是用來(lái)生成我們查數(shù)據(jù)庫(kù)的sql。它可以簡(jiǎn)化sql代碼的編寫,靈活、方便且易于維護(hù)2021-09-09詳解Idea 2019.2 安裝lombok插件失效問(wèn)題解決
這篇文章主要介紹了詳解Idea 2019.2 安裝lombok插件失效問(wèn)題解決,文中通過(guò)示例代碼介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友們下面隨著小編來(lái)一起學(xué)習(xí)學(xué)習(xí)吧2019-10-10Spring?代理?Bean?獲取不到原始?Bean?對(duì)象注解解決方法
這篇文章主要介紹了Spring?代理?Bean?獲取不到原始?Bean?對(duì)象注解解決方法,文章圍繞主題相關(guān)資料展開詳細(xì)介紹,需要的小伙伴可以參考一下2022-04-04使用 Spring Boot 實(shí)現(xiàn) WebSocket實(shí)時(shí)通信
本篇文章主要介紹了使用 Spring Boot 實(shí)現(xiàn) WebSocket實(shí)時(shí)通信,小編覺(jué)得挺不錯(cuò)的,現(xiàn)在分享給大家,也給大家做個(gè)參考。一起跟隨小編過(guò)來(lái)看看吧2017-10-10Java正則表達(dá)式如何匹配特定html標(biāo)簽內(nèi)的內(nèi)容
這篇文章主要給大家介紹了關(guān)于Java正則表達(dá)式如何匹配特定html標(biāo)簽內(nèi)的內(nèi)容的相關(guān)資料,文中通過(guò)示例代碼介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友們下面隨著小編來(lái)一起學(xué)習(xí)學(xué)習(xí)吧2020-09-09