通過String.intern()方法淺談堆中常量池
簡(jiǎn)介
String是我們最常用的一個(gè)類,和普通java類一樣其對(duì)象會(huì)存在java堆中。但是String類有其特殊之處,可以通過new方法生成,也可以通過帶引號(hào)的字符串常量直接賦值。在JDK7之前,字符串常量是存在永久帶Perm 區(qū)的,JDK7開始在將常量池遷移到堆中,這個(gè)變化也導(dǎo)致了String的新特性,下面我們慢慢進(jìn)行介紹。
String.intern()方法
簡(jiǎn)單的說,String.intern()方法的作用就是返回常量池中字符串對(duì)象,在對(duì)該方法進(jìn)行詳解之前,我們看幾個(gè)創(chuàng)建字符串對(duì)象的例子。以下說明及運(yùn)行結(jié)果都是以JDK8為java環(huán)境。
(1)直接賦值字符串常量
這種方式會(huì)判斷常量池中是否存在字符串常量,如果存在返回該常量對(duì)象,否則在常量池中創(chuàng)建常量對(duì)象并返回。
//在常量池中創(chuàng)建常量“abc”,s1,s2指向常量池中對(duì)象地址 String s1 = "abc"; String s2 = "abc"; System.out.println(s1 == s2);//true
(2)通過new關(guān)鍵字創(chuàng)建
這種方式會(huì)在堆上創(chuàng)建String對(duì)象,如果常量池中沒有該常量,將常量加入常量池中。
//在堆上創(chuàng)建對(duì)象S3,S4,常量池中創(chuàng)建對(duì)象“abc”
String s3 = new String("abc");
String s4 = new String("abc");
System.out.println(s3 == s4);//false
(3)字符串常量相加
這種方式如s5,會(huì)在常量池中創(chuàng)建"cd","ef","cdef"三個(gè)對(duì)象,s5指向常量池中的"cdef"對(duì)象。
String s5 = "cd" + "ef"; String s6 = "cdef"; System.out.println(s5==s6);//true
(4)兩個(gè)new的String對(duì)象相加
這種方式如s7,會(huì)在堆中創(chuàng)建三個(gè)對(duì)象"gh"對(duì)象,"lm"對(duì)象,以及"ghlm"對(duì)象,在常量池中創(chuàng)建對(duì)象"gh","lm"。
String s7 = new String("gh") + new String("lm");
String s8 = "ghlm";
System.out.println(s7==s8);//false
(5)字符串常量與new的String對(duì)象相加
這種方式如s9,會(huì)在堆中創(chuàng)建兩個(gè)對(duì)象“op”,“mnop”,并將字符串常量“op”, "mn"加到常量池中。
String s9 = "mn" + new String("op");
String s10 = "mnop";
System.out.println(s9==s10);//false
了解字符串常量的創(chuàng)建及其在內(nèi)存中的存儲(chǔ),我們看native方法intern()的作用:判斷String對(duì)象的常量值是否存在于常量池中,如果存在并且是常量池對(duì)象,返回該常量池對(duì)象;如果存在并且是指向堆中的對(duì)象,返回堆中對(duì)象地址;如果不存在,則將對(duì)象的引用復(fù)制到常量池,并返回該對(duì)象的引用。下面我們看幾條語句的運(yùn)行結(jié)果,第一個(gè)輸出之所以為true,
String s11 = new String("a") + new String("a");
s11.intern();//由于常量池中無“aa”,因此在常量池中建“aa”的引用,指向堆中的s11
String s12 = "aa";//s12指向常量池中的對(duì)象(該對(duì)象指向S11)
System.out.println(s11 == s12.intern());//true
String s13 = new String("b");
s13.intern();//常量池中已經(jīng)有“b”了,不做任何操作
String s14 = "b";
System.out.println(s13==s14.intern());//false
如果理解了以上運(yùn)行的結(jié)果,對(duì)intern()方法的左右就掌握的差不多了。那么久可以開始我們的主題,string pool,字符串常量池。
字符串常量池
字符串常量池是jvm為了減小內(nèi)存開銷而在創(chuàng)建字符串對(duì)象時(shí)的一個(gè)優(yōu)化,類似緩沖區(qū)。在hotspot中,字符串常量池是一個(gè)叫做StringTable的HashTable,默認(rèn)長(zhǎng)度是1009,在JDK7開始可以通過"-XX:StringTableSize=1009" 參數(shù)來設(shè)置,字符串常量池?cái)?shù)據(jù)可以被gc回收(在JDK6及其以前,字符串常量存在永久帶無法被gc回收,如果添加太多字符串常量到該區(qū)域,容易發(fā)生OOM)。由于字符串常量池是利用HashTable實(shí)現(xiàn),因此一定會(huì)發(fā)生hash碰撞。
jvm在這方面做了一定優(yōu)化,會(huì)根據(jù)hashTable的碰撞情況來決定是否做rehash,當(dāng)從這個(gè)StringTable里查找某個(gè)字符串是否存在,如果對(duì)其對(duì)應(yīng)的桶鏈表進(jìn)行遍歷,遍歷超過了100個(gè)節(jié)點(diǎn)還是沒有找到,那就會(huì)設(shè)置一個(gè)flag,讓下次進(jìn)入到safepoint的時(shí)候做一次rehash動(dòng)作,盡量減少碰撞的發(fā)生。當(dāng)然,在數(shù)據(jù)量比較大的情況下,這也無法從根本上解決問題,只能設(shè)置StringTableSize的值來緩解。
由于JDK7開始字符串常量池在堆中分布,所以young gc過程會(huì)掃描該區(qū)域,以保證處于新生代的String對(duì)象不會(huì)被回收掉,因此如果字符串常量區(qū)非常龐大會(huì)導(dǎo)致young gc過程掃描的時(shí)間也會(huì)變長(zhǎng)。但是,young gc階段并不會(huì)對(duì)字符串常量區(qū)進(jìn)行回收,具體回收階段是在Full gc或者CMS gc階段(題外話:我覺得full gc這個(gè)名字并不是很好,容易理解為對(duì)所有區(qū)域進(jìn)行回收,其實(shí)full GC是對(duì)老年代的STW的gc,full gc的次數(shù)是老年代gc的STW次數(shù),時(shí)間是老年代STW的總時(shí)間)。
以上就是本文的全部?jī)?nèi)容,希望對(duì)大家的學(xué)習(xí)有所幫助,也希望大家多多支持腳本之家。
相關(guān)文章
SpringCloud的Gateway網(wǎng)關(guān)詳解
這篇文章主要介紹了SpringCloud的Gateway網(wǎng)關(guān)詳解,Gateway 是 Spring Cloud 官方推出的一個(gè)基于 Spring 5、Spring Boot 2 和 Project Reactor 的 API 網(wǎng)關(guān)實(shí)現(xiàn),本文將介紹 Spring Cloud Gateway 的基本概念、核心組件以及如何配置和使用它,需要的朋友可以參考下2023-09-09
Android Studio更改項(xiàng)目使用的JDK(詳細(xì)步驟)
本文介紹了如何在Android Studio中修改Gradle和JDK的配置步驟,包括打開設(shè)置、進(jìn)入Gradle設(shè)置、修改JDK路徑、保存并生效等,感興趣的朋友跟隨小編一起看看吧2024-11-11
netty對(duì)proxy protocol代理協(xié)議的支持詳解
這篇文章主要為大家介紹了netty對(duì)proxy protoco代理協(xié)議的支持詳解,有需要的朋友可以借鑒參考下,希望能夠有所幫助,祝大家多多進(jìn)步,早日升職加薪2022-07-07
mybatis的動(dòng)態(tài)SQL以及連接池詳解
這篇文章主要介紹了mybatis的動(dòng)態(tài)SQL以及連接池詳解,具有很好的參考價(jià)值,希望對(duì)大家有所幫助。如有錯(cuò)誤或未考慮完全的地方,望不吝賜教2022-02-02
java實(shí)現(xiàn)字符串轉(zhuǎn)String數(shù)組的方法示例
這篇文章主要介紹了java實(shí)現(xiàn)字符串轉(zhuǎn)String數(shù)組的方法,涉及java字符串的遍歷、分割、轉(zhuǎn)換等相關(guān)操作技巧,需要的朋友可以參考下2017-10-10
通俗易懂的Java常見限流算法具體實(shí)現(xiàn)
這篇文章主要介紹了Java常見限流算法具體實(shí)現(xiàn)的相關(guān)資料,包括漏桶算法、令牌桶算法、Nginx限流和Redis+Lua限流的實(shí)現(xiàn)原理和具體步驟,并比較了它們的優(yōu)點(diǎn)和缺點(diǎn),需要的朋友可以參考下2025-02-02
Java中?springcloud.openfeign應(yīng)用案例解析
使用OpenFeign能讓編寫Web?Service客戶端更加簡(jiǎn)單,使用時(shí)只需定義服務(wù)接口,然后在上面添加注解,OpenFeign也支持可拔插式的編碼和解碼器,這篇文章主要介紹了Java中?springcloud.openfeign應(yīng)用案例解析,需要的朋友可以參考下2024-06-06

