淺談為什么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.net.http.HttpClient使用示例解析
這篇文章主要為大家介紹了java.net.http.HttpClient使用示例解析,有需要的朋友可以借鑒參考下,希望能夠有所幫助,祝大家多多進步,早日升職加薪2023-08-08
Mybatis-plus 批量插入太慢的問題解決(提升插入性能)
公司使用的Mybatis-Plus操作SQL,用過Mybatis-Plus的小伙伴一定知道他有很多API提供給我們使用,但是批量插入大數(shù)據(jù)太慢應(yīng)該怎么解決,本文就詳細的介紹一下,感興趣的可以了解一下2021-11-11
Java使用Knife4j優(yōu)化Swagger接口文檔的操作步驟
在現(xiàn)代微服務(wù)開發(fā)中,接口文檔的質(zhì)量直接影響了前后端協(xié)作效率,Swagger 作為一個主流的接口文檔工具,雖然功能強大,但其默認界面和部分功能在實際使用中略顯不足,而 Knife4j 的出現(xiàn)為我們提供了一種增強的選擇,本篇文章將詳細介紹如何在項目中集成和使用 Knife4j2024-12-12
聊聊spring @Transactional 事務(wù)無法使用的可能原因
這篇文章主要介紹了spring @Transactional 事務(wù)無法使用的可能原因,具有很好的參考價值,希望對大家有所幫助。如有錯誤或未考慮完全的地方,望不吝賜教2021-07-07
SpringBoot使用Sa-Token實現(xiàn)路徑攔截和特定接口放行
這篇文章主要介紹了SpringBoot使用Sa-Token實現(xiàn)路徑攔截和特定接口放行,文中通過代碼示例講解的非常詳細,對大家的學(xué)習(xí)或工作有一定的幫助,需要的朋友可以參考下2024-06-06
@ConfigurationProperties在IDEA中出現(xiàn)紅色波浪線問題解決方法
本文介紹了在Springboot項目中,當@ConfigurationProperties注解出現(xiàn)紅色波浪線時的解決方法,文中有詳細的解決方案供大家參考,需要的朋友可以參考下2024-09-09

