欧美一区二区三区老妇人-欧美做爰猛烈大尺度电-99久久夜色精品国产亚洲a-亚洲福利视频一区二区

通過String.intern()方法淺談堆中常量池

簡(jiǎn)介

創(chuàng)新互聯(lián)公司專注于荷塘企業(yè)網(wǎng)站建設(shè),響應(yīng)式網(wǎng)站開發(fā),成都商城網(wǎng)站開發(fā)。荷塘網(wǎng)站建設(shè)公司,為荷塘等地區(qū)提供建站服務(wù)。全流程按需網(wǎng)站設(shè)計(jì),專業(yè)設(shè)計(jì),全程項(xiàng)目跟蹤,創(chuàng)新互聯(lián)公司專業(yè)和態(tài)度為您提供的服務(wù)

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í)有所幫助,也希望大家多多支持創(chuàng)新互聯(lián)。

新聞名稱:通過String.intern()方法淺談堆中常量池
鏈接URL:http://www.chinadenli.net/article44/isjgee.html

成都網(wǎng)站建設(shè)公司_創(chuàng)新互聯(lián),為您提供虛擬主機(jī)標(biāo)簽優(yōu)化網(wǎng)站維護(hù)網(wǎng)站制作網(wǎng)站導(dǎo)航Google

廣告

聲明:本網(wǎng)站發(fā)布的內(nèi)容(圖片、視頻和文字)以用戶投稿、用戶轉(zhuǎn)載內(nèi)容為主,如果涉及侵權(quán)請(qǐng)盡快告知,我們將會(huì)在第一時(shí)間刪除。文章觀點(diǎn)不代表本網(wǎng)站立場(chǎng),如需處理請(qǐng)聯(lián)系客服。電話:028-86922220;郵箱:631063699@qq.com。內(nèi)容未經(jīng)允許不得轉(zhuǎn)載,或轉(zhuǎn)載時(shí)需注明來源: 創(chuàng)新互聯(lián)

成都網(wǎng)站建設(shè)公司