Java String的intern方法使用場(chǎng)景示例
在講intern方法前,我們先簡(jiǎn)單回顧下Java中常量池的分類(lèi)。
常量池的分類(lèi)
Java中常量池可以分為Class常量池、運(yùn)行時(shí)常量池和字符串常量池。
1. Class文件常量池
在Class文件中除了有類(lèi)的版本、字段、方法、接口等描述信息外,還有一項(xiàng)信息是常量池(Constant Pool Table),用于存放編譯期生成的各種字面量和符號(hào)引用。
所謂字面量類(lèi)似與我們平常說(shuō)的常量,主要包括以下兩種
- 文本字符串,例如String a = "aa"。其中"aa"就是字面量。
- 被final修飾的變量。
符號(hào)引用包括以下形式:
- 類(lèi)和接口和全限定名:例如對(duì)于String這個(gè)類(lèi),它的全限定名就是java/lang/String。
- 字段的名稱(chēng)和描述符:所謂字段就是類(lèi)或者接口中聲明的變量,包括類(lèi)級(jí)別變量和實(shí)例級(jí)的變量。
- 方法的名稱(chēng)和描述符:所謂描述符就相當(dāng)于方法的參數(shù)類(lèi)型+返回值類(lèi)型。
2. 運(yùn)行時(shí)常量池
我們知道類(lèi)加載器會(huì)加載對(duì)應(yīng)的Class文件,上面介紹的Class文件常量池中的數(shù)據(jù),會(huì)在類(lèi)加載后進(jìn)入方法區(qū)中的運(yùn)行時(shí)常量池。運(yùn)行時(shí)常量池是全局共享的,多個(gè)類(lèi)共用一個(gè)運(yùn)行時(shí)常量池。運(yùn)行時(shí)常量池存在于方法區(qū)中。
3. 字符串常量池
看名字我們就可以知道字符串常量池是用來(lái)存放字符串的,也就是說(shuō)Class文件常量池中的文本字符串會(huì)在類(lèi)加載時(shí)進(jìn)入字符串常量池。
那字符串常量池和運(yùn)行時(shí)常量池是什么關(guān)系呢?上面我們說(shuō)Class文件常量池中的字面量會(huì)在類(lèi)加載后進(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:true
s1==s2:false
s1==s2:true
s5==s6:false
s6==s8:true
s7==s8:false
intern 方法使用場(chǎng)景
我們來(lái)看下面這個(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ì)象的空間。我寫(xiě)了一個(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é)果如下:
... 93889 93890 Exception in thread "main" java.lang.OutOfMemoryError: GC overhead limit exceeded at 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();
}
}
99997
99998
99999
success...
順利執(zhí)行完成。
以上就是Java String的intern方法使用場(chǎng)景示例的詳細(xì)內(nèi)容,更多關(guān)于Java String的intern方法的資料請(qǐng)關(guān)注腳本之家其它相關(guān)文章!
相關(guān)文章
在Linux系統(tǒng)上升級(jí)Java版本的兩種方法步驟
由于項(xiàng)目升級(jí),需要將JDK7升級(jí)到JDK8,升級(jí)JDK的同時(shí)也要升級(jí)一些其他的版本,下面這篇文章主要給大家介紹了關(guān)于在Linux系統(tǒng)上升級(jí)Java版本的兩種方法步驟,需要的朋友可以參考下2024-09-09
Spring中的ThreadPoolTaskExecutor線(xiàn)程池使用詳解
這篇文章主要介紹了Spring中的ThreadPoolTaskExecutor線(xiàn)程池使用詳解,ThreadPoolTaskExecutor 是 Spring框架提供的一個(gè)線(xiàn)程池實(shí)現(xiàn),用于管理和執(zhí)行多線(xiàn)程任務(wù),它是TaskExecutor接口的實(shí)現(xiàn),提供了在 Spring 應(yīng)用程序中創(chuàng)建和配置線(xiàn)程池的便捷方式,需要的朋友可以參考下2024-01-01
Java 8 Stream 的終極技巧——Collectors 功能與操作方法詳解
這篇文章主要介紹了Java 8 Stream Collectors 功能與操作方法,結(jié)合實(shí)例形式詳細(xì)分析了Java 8 Stream Collectors 功能、操作方法及相關(guān)注意事項(xiàng),需要的朋友可以參考下2020-05-05
Spring Boot 集成Shiro的多realm配置過(guò)程
這篇文章主要介紹了Spring Boot 集成Shiro的多realm配置,本文給大家介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或工作具有一定的參考借鑒價(jià)值,需要的朋友可以參考下2020-10-10
Ajax實(shí)現(xiàn)省市區(qū)三級(jí)聯(lián)動(dòng)
這篇文章主要為大家詳細(xì)介紹了jQuery ajax實(shí)現(xiàn)省市縣三級(jí)聯(lián)動(dòng)的相關(guān)資料,具有一定的參考價(jià)值,感興趣的小伙伴們可以參考一下,希望能幫助到你2021-07-07
手把手教你實(shí)現(xiàn)idea中配置國(guó)內(nèi)源
idea的國(guó)內(nèi)源配置十分重要,能夠提升程序開(kāi)發(fā)的效率而且也是減少bug的一種有效防范,本文就來(lái)介紹一下idea中配置國(guó)內(nèi)源,具有一定的參考價(jià)值,感興趣的可以了解一下2023-07-07
java中ConcurrentHashMap的讀操作為什么不需要加鎖
ConcurrentHashMap完全允許多個(gè)讀操作并發(fā)進(jìn)行,讀操作并不需要加鎖。所以下面這篇文章主要給大家介紹了關(guān)于java中ConcurrentHashMap的讀操作為什么不需要加鎖的相關(guān)資料,需要的朋友可以參考下2018-10-10
httpclient connect連接請(qǐng)求方法源碼解讀
這篇文章主要為大家介紹了httpclient connect連接請(qǐng)求方法解讀,有需要的朋友可以借鑒參考下,希望能夠有所幫助,祝大家多多進(jìn)步,早日升職加薪2023-11-11

