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