String中intern方法的使用場景詳解
在講intern方法前,我們先簡單回顧下Java中常量池的分類。
常量池的分類#
Java中常量池可以分為Class常量池、運行時常量池和字符串常量池。
1. Class文件常量池
在Class文件中除了有類的版本、字段、方法、接口等描述信息外,還有一項信息是常量池(Constant Pool Table),用于存放編譯期生成的各種字面量和符號引用。
所謂字面量類似與我們平常說的常量,主要包括以下兩種
- 文本字符串,例如String a = "aa"。其中"aa"就是字面量。
- 被final修飾的變量。
符號引用包括以下形式:
- 類和接口和全限定名:例如對于String這個類,它的全限定名就是java/lang/String。
- 字段的名稱和描述符:所謂字段就是類或者接口中聲明的變量,包括類級別變量和實例級的變量。
- 方法的名稱和描述符:所謂描述符就相當于方法的參數(shù)類型+返回值類型。
2. 運行時常量池
我們知道類加載器會加載對應(yīng)的Class文件,上面介紹的Class文件常量池中的數(shù)據(jù),會在類加載后進入方法區(qū)中的運行時常量池。運行時常量池是全局共享的,多個類共用一個運行時常量池。運行時常量池存在于方法區(qū)中。
3. 字符串常量池
看名字我們就可以知道字符串常量池是用來存放字符串的,也就是說Class文件常量池中的文本字符串會在類加載時進入字符串常量池。
那字符串常量池和運行時常量池是什么關(guān)系呢?上面我們說Class文件常量池中的字面量會在類加載后進入運行時常量池,其中字面量中也包括文本字符串,從這段文字我們可以知道字符串常量池存在于運行時常量池中,也就存在于方法區(qū)中。
但是到了JDK1.7時,字符串常量池被移出了方法區(qū),轉(zhuǎn)移到了堆里了。另外需要我們重點注意的是:字符串常量池中存放的并不是字符串本身,而是字符串對象的引用。
程序運行時,除非手動向常量池中添加常量(比如調(diào)用intern方法),否則jvm不會自動添加常量到常量池。
String 的 intern 方法#
String 方法的作用是:判斷字符串常量池中是否存在一個引用,這個引用指向的字符串對象和當前對象相等(使用 equals 方法判斷相等),如果存在直接返回這個引用,如果不存在則創(chuàng)建一個字符串對象并將其引用存入字符串常量池。
下面舉個列子幫助加深理解。
//代碼基于JDK 8 //s1指向字符串常量池中的"自由之路" String s1 = "自由之路"; //s2也指向字符串常量池中的"自由之路" String s2 = "自由之路"; //s3指向堆中的某個對象 String s3 = new String("自由之路"); //因為字符串常量池中已經(jīng)存在"自由之路"的引用,直接返回這個引用 String s4 = s3.intern(); //創(chuàng)建一個字符串對象 String s5 = new String("ddd"); //常量池中不存在指向"ddd"的引用,創(chuàng)建一個"ddd"對象,并將其引用存入常量池 String s6 = s5.intern(); //創(chuàng)建一個字符串對象 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 方法使用場景#
我們來看下面這個方法。
public class Person{ String name; public void setName(String name) { this.name = name } }
假如現(xiàn)在的Person對象都叫小明,那么這些Person對象都會引用一個不同的字符串對象。
如果我們改進下這個方法:
public class Person{ String name; public void setName(String name) { this.name = name.intern(); } }
那么對象的引用結(jié)構(gòu)如下圖所示:
這樣明顯可以節(jié)省多個字符串對象的空間。我寫了一個測試程序:
public class JavaTest { public static void main(String[] args) throws Exception { //一個很大的字符串 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多個對象時已經(jīng)報OutOfMemoryError錯誤了。
下面調(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í)行完成。
總結(jié)
到此這篇關(guān)于String中intern方法的使用場景的文章就介紹到這了,更多相關(guān)String intern方法使用場景內(nèi)容請搜索腳本之家以前的文章或繼續(xù)瀏覽下面的相關(guān)文章希望大家以后多多支持腳本之家!
相關(guān)文章
Java中LocalDate日期格式轉(zhuǎn)換(使用系統(tǒng)時區(qū))
本文主要介紹了Java中LocalDate日期格式轉(zhuǎn)換(使用系統(tǒng)時區(qū)),文中通過示例代碼介紹的非常詳細,對大家的學習或者工作具有一定的參考學習價值,需要的朋友們下面隨著小編來一起學習學習吧2007-02-02Spring Boot Admin(監(jiān)控工具)的使用
今天我們將會講解一個優(yōu)秀的監(jiān)控工具Spring Boot Admin。 它采用圖形化的界面,讓我們的Spring Boot管理更加簡單,需要的朋友可以參考下2020-02-02聊聊Spring循環(huán)依賴三級緩存是否可以減少為二級緩存的情況
這篇文章主要介紹了聊聊Spring循環(huán)依賴三級緩存是否可以減少為二級緩存的情況,具有很好的參考價值,希望對大家有所幫助。一起跟隨小編過來看看吧2021-02-02SpringCloud?openfeign聲明式服務(wù)調(diào)用實現(xiàn)方法介紹
在springcloud中,openfeign是取代了feign作為負載均衡組件的,feign最早是netflix提供的,他是一個輕量級的支持RESTful的http服務(wù)調(diào)用框架,內(nèi)置了ribbon,而ribbon可以提供負載均衡機制,因此feign可以作為一個負載均衡的遠程服務(wù)調(diào)用框架使用2022-12-12Spring Boot2.0使用Spring Security的示例代碼
這篇文章主要介紹了Spring Boot2.0使用Spring Security的示例代碼,小編覺得挺不錯的,現(xiàn)在分享給大家,也給大家做個參考。一起跟隨小編過來看看吧2018-08-08Java實現(xiàn)優(yōu)雅的參數(shù)校驗方法詳解
這篇文章主要為大家詳細介紹了Java語言如何實現(xiàn)優(yōu)雅的參數(shù)校驗,文中的示例代碼講解詳細,對我們學習Java有一定是幫助,需要的可以參考一下2022-06-06