常量池之字符串常量池String.intern()
運(yùn)行時(shí)常量池是方法區(qū)(PermGen)的一部分。
需要提前了解:
1. JVM內(nèi)存模型。
2. JAVA對(duì)象在JVM中內(nèi)存分配
常量池的好處
常量池是為了避免頻繁的創(chuàng)建和銷毀對(duì)象而影響系統(tǒng)性能,其實(shí)現(xiàn)了對(duì)象的共享。
- Java的自動(dòng)裝箱中其實(shí)就使用到了運(yùn)行時(shí)常量池。詳見:Java 自動(dòng)裝箱與拆箱的實(shí)現(xiàn)原理
- 還有字符串常量池。
字符串進(jìn)入到常量池的兩種方法:
1. new String()的實(shí)例調(diào)用intern()方法。
????執(zhí)行intern()方法時(shí),若常量池中不存在等值的字符串,JVM就會(huì)在常量池中 創(chuàng)建一個(gè)等值的字符串,然后返回該字符串的引用。
2. “”(引號(hào))引起來的內(nèi)容(字面量)。
????引號(hào)引起來的字符串,首先從常量池中查找是否存在此字符串,如果不存在則在常量池中添加此字符串對(duì)象,然后引用此字符串對(duì)象。如果存在,則直接引用此字符串。
重要提示:虛擬機(jī)啟動(dòng)時(shí)常量池中就存在“java”字符串實(shí)例,下面代碼中s2調(diào)用intern()方法時(shí),只是返回常量池中“java”實(shí)例的引用,而沒有添加“java”實(shí)例。
先看下,下面的代碼
public class Test {public static void main(String[] args) {String s1 = new StringBuilder().append("aa").append("bb").toString();System.out.println(s1.intern() == s1);String s2 = new StringBuilder().append("ja").append("va").toString();System.out.println(s2.intern() == s2); } }輸出結(jié)果如下:
JDK6
false
false
JDK7、JDK8
true
false
為什么不同版本的JDK輸出的結(jié)果還不一樣呢?帶著疑問我們分別分析下JDK1.6 和JDK1.7-JDK1.8的常量池內(nèi)存模型。
JDK1.6 常量池內(nèi)存模型
如圖中,我們可以看到常量池位于方法區(qū)(PermGen),常量池中引用的字符串實(shí)例也在常量池中。
1、通過調(diào)用intern()方法,會(huì)在常量池中生成一個(gè)相同字符串的對(duì)象
2、“”內(nèi)的字符串都會(huì)添加到常量池中,相當(dāng)于引用的方法區(qū)中的字符串對(duì)象。
根據(jù)上圖內(nèi)存模型然后我們?cè)诜治鱿麓a:
String s1 = new StringBuilder().append(“aa”).append(“bb”).toString();
System.out.println(s1.intern() == s1);
s1生成的對(duì)象在堆中,而s1.intern()的對(duì)象在常量池中,所以返回false。
String s2 = new StringBuilder().append(“ja”).append(“va”).toString();
System.out.println(s2.intern() == s2);
s2生成的對(duì)象在堆中,而s2.intern()的對(duì)象也肯定在常量池中,所以也返回false。
JDK1.7-JDK1.8常量池內(nèi)存模型
如圖中,我們可以看到常量池位于方法區(qū)(PermGen),但常量池引用的字符串實(shí)例在堆中。(區(qū)別在這)
1. 通過調(diào)用intern()方法,會(huì)在常量池添加一個(gè)此字符串實(shí)例的引用,(前提:常量池中沒有相同內(nèi)容的字符串)。
2. “”內(nèi)的字符串實(shí)例引用會(huì)添加到常量池中(前提:常量池中沒有相同內(nèi)容的字符串),如果常量池中存在,則引用常量池中的對(duì)象(防止重復(fù)創(chuàng)建對(duì)象)。
根據(jù)上圖內(nèi)存模型然后我們?cè)诜治鱿麓a:
String s1 = new StringBuilder().append(“aa”).append(“bb”).toString();
System.out.println(s1.intern() == s1);
s1生成的對(duì)象在堆中,此時(shí)常量池中沒有跟s1內(nèi)容相同的字符串,所以在調(diào)用intern方法時(shí),會(huì)在常量池中添加此對(duì)象的引用,所以返回為true。
String s2 = new StringBuilder().append(“ja”).append(“va”).toString();
System.out.println(s2.intern() == s2);
s2生成的對(duì)象在堆中,而此時(shí)常量池中已經(jīng)有一個(gè)跟s2內(nèi)容相同的字符串常量,當(dāng)s2調(diào)用intern方法時(shí),返回常量池中已經(jīng)存在的實(shí)例(相當(dāng)于堆中有兩個(gè)相同內(nèi)容的實(shí)例:一個(gè)是new 出來的,一個(gè)是常量池中的)所以返回的結(jié)果為false。
本人簡(jiǎn)書blog地址:http://www.jianshu.com/u/1f0067e24ff8????
點(diǎn)擊這里快速進(jìn)入簡(jiǎn)書
總結(jié)
以上是生活随笔為你收集整理的常量池之字符串常量池String.intern()的全部?jī)?nèi)容,希望文章能夠幫你解決所遇到的問題。
- 上一篇: ConcurrentHashMap 原理
- 下一篇: java中的CAS和原子类的实现