Java String的intern方法使用場(chǎng)景示例
在講intern方法前,我們先簡(jiǎn)單回顧下Java中常量池的分類。
常量池的分類
Java中常量池可以分為Class常量池、運(yùn)行時(shí)常量池和字符串常量池。
1. Class文件常量池
在Class文件中除了有類的版本、字段、方法、接口等描述信息外,還有一項(xiàng)信息是常量池(Constant Pool Table),用于存放編譯期生成的各種字面量和符號(hào)引用。
所謂字面量類似與我們平常說的常量,主要包括以下兩種
文本字符串,例如String a = 'aa'。其中'aa'就是字面量。 被final修飾的變量。符號(hào)引用包括以下形式:
類和接口和全限定名:例如對(duì)于String這個(gè)類,它的全限定名就是java/lang/String。 字段的名稱和描述符:所謂字段就是類或者接口中聲明的變量,包括類級(jí)別變量和實(shí)例級(jí)的變量。 方法的名稱和描述符:所謂描述符就相當(dāng)于方法的參數(shù)類型+返回值類型。2. 運(yùn)行時(shí)常量池
我們知道類加載器會(huì)加載對(duì)應(yīng)的Class文件,上面介紹的Class文件常量池中的數(shù)據(jù),會(huì)在類加載后進(jìn)入方法區(qū)中的運(yùn)行時(shí)常量池。運(yùn)行時(shí)常量池是全局共享的,多個(gè)類共用一個(gè)運(yùn)行時(shí)常量池。運(yùn)行時(shí)常量池存在于方法區(qū)中。
3. 字符串常量池
看名字我們就可以知道字符串常量池是用來存放字符串的,也就是說Class文件常量池中的文本字符串會(huì)在類加載時(shí)進(jìn)入字符串常量池。
那字符串常量池和運(yùn)行時(shí)常量池是什么關(guān)系呢?上面我們說Class文件常量池中的字面量會(huì)在類加載后進(jìn)入運(yùn)行時(shí)常量池,其中字面量中也包括文本字符串,從這段文字我們可以知道字符串常量池存在于運(yùn)行時(shí)常量池中,也就存在于方法區(qū)中。
但是到了JDK1.7時(shí),字符串常量池被移出了方法區(qū),轉(zhuǎn)移到了堆里了。另外需要我們重點(diǎn)注意的是:字符串常量池中存放的并不是字符串本身,而是字符串對(duì)象的引用。
程序運(yùn)行時(shí),除非手動(dòng)向常量池中添加常量(比如調(diào)用intern方法),否則jvm不會(huì)自動(dòng)添加常量到常量池。
String 的 intern 方法
String 方法的作用是:判斷字符串常量池中是否存在一個(gè)引用,這個(gè)引用指向的字符串對(duì)象和當(dāng)前對(duì)象相等(使用 equals 方法判斷相等),如果存在直接返回這個(gè)引用,如果不存在則創(chuàng)建一個(gè)字符串對(duì)象并將其引用存入字符串常量池。
下面舉個(gè)列子幫助加深理解。
//代碼基于JDK 8//s1指向字符串常量池中的'自由之路'String s1 = '自由之路';//s2也指向字符串常量池中的'自由之路'String s2 = '自由之路';//s3指向堆中的某個(gè)對(duì)象String s3 = new String('自由之路');//因?yàn)樽址A砍刂幸呀?jīng)存在'自由之路'的引用,直接返回這個(gè)引用String s4 = s3.intern();//創(chuàng)建一個(gè)字符串對(duì)象String s5 = new String('ddd');//常量池中不存在指向'ddd'的引用,創(chuàng)建一個(gè)'ddd'對(duì)象,并將其引用存入常量池String s6 = s5.intern();//創(chuàng)建一個(gè)字符串對(duì)象String s7 = new String('ddd');//常量池中存在指向'ddd'的引用,直接返回String s8 = s7.intern();System.out.println('s1==s2:'+(s1==s2));System.out.println('s1==s3:'+(s1==s3));System.out.println('s1==s4:'+(s1==s4));System.out.println('s5==s6:'+(s5==s6));System.out.println('s6==s8:'+(s6==s8));System.out.println('s7==s8:'+(s7==s8));
返回的結(jié)果如下:
s1==s2:trues1==s2:falses1==s2:trues5==s6:falses6==s8:trues7==s8:false
intern 方法使用場(chǎng)景
我們來看下面這個(gè)方法。
public class Person{ String name; public void setName(String name) { this.name = name }}
假如現(xiàn)在的Person對(duì)象都叫小明,那么這些Person對(duì)象都會(huì)引用一個(gè)不同的字符串對(duì)象。
如果我們改進(jìn)下這個(gè)方法:
public class Person{ String name; public void setName(String name) { this.name = name.intern(); }}
那么對(duì)象的引用結(jié)構(gòu)如下圖所示
這樣明顯可以節(jié)省多個(gè)字符串對(duì)象的空間。我寫了一個(gè)測(cè)試程序:
public class JavaTest { public static void main(String[] args) throws Exception { //一個(gè)很大的字符串 String s = 'c...c'; List<Person> personList = new ArrayList<>(); int count = 100000; for (int i = 0; i < count; i++) { Person p = new Person(); p.setName(new String(s)); //防止垃圾回收 personList.add(p); System.out.println(i); } System.out.println('success...'); } public static class Person{ private String name; public void setName(String name) { this.name = name; } }}
為了讓程序快速將內(nèi)存耗盡,我這邊將內(nèi)存設(shè)置成5M。
-Xms5m -Xmx5m
結(jié)果如下:
...9388993890Exception in thread 'main' java.lang.OutOfMemoryError: GC overhead limit exceededat com.csx.demo.spring.boot.util.JavaTest.main(JavaTest.java:15)
創(chuàng)建9w多個(gè)對(duì)象時(shí)已經(jīng)報(bào)OutOfMemoryError錯(cuò)誤了。
下面調(diào)整下 Person 的 set 方法,再執(zhí)行下。
public static class Person{ private String name; public void setName(String name) { this.name = name.intern(); }}
999979999899999success...
順利執(zhí)行完成。
以上就是Java String的intern方法使用場(chǎng)景示例的詳細(xì)內(nèi)容,更多關(guān)于Java String的intern方法的資料請(qǐng)關(guān)注好吧啦網(wǎng)其它相關(guān)文章!
相關(guān)文章:
1. WML教程之文本框控件Input2. 詳解CSS偽元素的妙用單標(biāo)簽之美3. XML入門的常見問題(三)4. 利用CSS3新特性創(chuàng)建透明邊框三角5. Vue3獲取DOM節(jié)點(diǎn)的3種方式實(shí)例6. 不要在HTML中濫用div7. 多級(jí)聯(lián)動(dòng)下拉選擇框,動(dòng)態(tài)獲取下一級(jí)8. CSS3實(shí)例分享之多重背景的實(shí)現(xiàn)(Multiple backgrounds)9. vue實(shí)現(xiàn)將自己網(wǎng)站(h5鏈接)分享到微信中形成小卡片的超詳細(xì)教程10. 前端html+css實(shí)現(xiàn)動(dòng)態(tài)生日快樂代碼
