日韩性视频-久久久蜜桃-www中文字幕-在线中文字幕av-亚洲欧美一区二区三区四区-撸久久-香蕉视频一区-久久无码精品丰满人妻-国产高潮av-激情福利社-日韩av网址大全-国产精品久久999-日本五十路在线-性欧美在线-久久99精品波多结衣一区-男女午夜免费视频-黑人极品ⅴideos精品欧美棵-人人妻人人澡人人爽精品欧美一区-日韩一区在线看-欧美a级在线免费观看

歡迎訪問 生活随笔!

生活随笔

當前位置: 首頁 > 编程资源 > 编程问答 >内容正文

编程问答

常量池之字符串常量池String.intern()

發布時間:2024/9/30 编程问答 31 豆豆
生活随笔 收集整理的這篇文章主要介紹了 常量池之字符串常量池String.intern() 小編覺得挺不錯的,現在分享給大家,幫大家做個參考.

運行時常量池是方法區(PermGen)的一部分。

需要提前了解:
1. JVM內存模型
2. JAVA對象在JVM中內存分配

常量池的好處

常量池是為了避免頻繁的創建和銷毀對象而影響系統性能,其實現了對象的共享。
- Java的自動裝箱中其實就使用到了運行時常量池。詳見:Java 自動裝箱與拆箱的實現原理
- 還有字符串常量池。

字符串進入到常量池的兩種方法:

1. new String()的實例調用intern()方法。
????執行intern()方法時,若常量池中不存在等值的字符串,JVM就會在常量池中 創建一個等值的字符串,然后返回該字符串的引用。
2. “”(引號)引起來的內容(字面量)。
????引號引起來的字符串,首先從常量池中查找是否存在此字符串,如果不存在則在常量池中添加此字符串對象,然后引用此字符串對象。如果存在,則直接引用此字符串。

重要提示:虛擬機啟動時常量池中就存在“java”字符串實例,下面代碼中s2調用intern()方法時,只是返回常量池中“java”實例的引用,而沒有添加“java”實例。

先看下,下面的代碼

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); } }

輸出結果如下:

JDK6

false
false

JDK7、JDK8

true
false

為什么不同版本的JDK輸出的結果還不一樣呢?帶著疑問我們分別分析下JDK1.6 和JDK1.7-JDK1.8的常量池內存模型。

JDK1.6 常量池內存模型

如圖中,我們可以看到常量池位于方法區(PermGen),常量池中引用的字符串實例也在常量池中。
1、通過調用intern()方法,會在常量池中生成一個相同字符串的對象
2、“”內的字符串都會添加到常量池中,相當于引用的方法區中的字符串對象。

根據上圖內存模型然后我們在分析下代碼:

String s1 = new StringBuilder().append(“aa”).append(“bb”).toString();
System.out.println(s1.intern() == s1);
s1生成的對象在堆中,而s1.intern()的對象在常量池中,所以返回false。

String s2 = new StringBuilder().append(“ja”).append(“va”).toString();
System.out.println(s2.intern() == s2);
s2生成的對象在堆中,而s2.intern()的對象也肯定在常量池中,所以也返回false。

JDK1.7-JDK1.8常量池內存模型

如圖中,我們可以看到常量池位于方法區(PermGen),但常量池引用的字符串實例在中。(區別在這)
1. 通過調用intern()方法,會在常量池添加一個此字符串實例的引用,(前提:常量池中沒有相同內容的字符串)。
2. “”內的字符串實例引用會添加到常量池中(前提:常量池中沒有相同內容的字符串),如果常量池中存在,則引用常量池中的對象(防止重復創建對象)。

根據上圖內存模型然后我們在分析下代碼:
String s1 = new StringBuilder().append(“aa”).append(“bb”).toString();
System.out.println(s1.intern() == s1);
s1生成的對象在堆中,此時常量池中沒有跟s1內容相同的字符串,所以在調用intern方法時,會在常量池中添加此對象的引用,所以返回為true。

String s2 = new StringBuilder().append(“ja”).append(“va”).toString();
System.out.println(s2.intern() == s2);
s2生成的對象在堆中,而此時常量池中已經有一個跟s2內容相同的字符串常量,當s2調用intern方法時,返回常量池中已經存在的實例(相當于堆中有兩個相同內容的實例:一個是new 出來的,一個是常量池中的)所以返回的結果為false。

本人簡書blog地址:http://www.jianshu.com/u/1f0067e24ff8????
點擊這里快速進入簡書

總結

以上是生活随笔為你收集整理的常量池之字符串常量池String.intern()的全部內容,希望文章能夠幫你解決所遇到的問題。

如果覺得生活随笔網站內容還不錯,歡迎將生活随笔推薦給好友。