国产探花免费观看_亚洲丰满少妇自慰呻吟_97日韩有码在线_资源在线日韩欧美_一区二区精品毛片,辰东完美世界有声小说,欢乐颂第一季,yy玄幻小说排行榜完本

首頁 > 學院 > 開發設計 > 正文

String創建方式內存終極分析

2019-11-08 01:35:43
字體:
來源:轉載
供稿:網友

string內存終極解析 猜猜下面代碼的運行結果?

public class TestString { public static void m1() { String a = "a1"; String b = "a" + 1; System.out.PRintln(a == b); } public static void m2() { String a = "ab"; String bb = "b"; String b = "a" + bb; System.out.println(a == b); } public static void m3() { String a = "ab"; final String bb = "b"; String b = "a" + bb; System.out.println(a == b); } public static void m4() { String a = "ab"; final String bb = getBB(); String b = "a" + bb; System.out.println(a == b); } private static String getBB() { return "b"; } private static String a = "ab"; public static void m5() { String s1 = "a"; String s2 = "b"; String s = s1 + s2; System.out.println(s == a); System.out.println(s.intern() == a); } private static String ab = new String("ab"); public static void m6() { String s1 = "a"; String s2 = "b"; String s = s1 + s2; System.out.println(s == ab); System.out.println(s.intern() == ab); System.out.println(s.intern() == ab.intern()); } private static void m7() { String s1 = "a"; String s2 = new String("a"); s2.intern(); System.out.println(s1 == s2); s2 = s2.intern(); System.out.println(s1 == s2); } public static void main(String[] args) { m1(); m2(); m3(); m4(); m5(); m6(); m7(); }}

如果能不含糊的說出運行結果,那你不用往下看了

概念:

要想解釋清楚原理,首先要明確幾個概念:

String: String對象是一種特殊的對象。String類是一個不可變的類(final修飾的類)。也就說,String對象一旦創建就不允許修改。String池:java用于保存String,在編譯期已確定的,已編譯的class文件中的一份可擴充數據。為了提高效率Java引用了字符串池的概念,用于維護java中String的一塊獨立內存,不同于堆內存和棧內存。Java中==比較: 通俗的說,==比較內存地址,如果內存地址相同,才返回true。編譯期和運行期: 在編譯期能確定的String存儲在String池中,不能確定的在運行期存在java堆中。String.intern()方法: 官方解釋Returns a canonical representation for the string object.(返回字符串對象的規范化表示形式。)簡單點說就是返回String池的字符串。

創建String的三種方式:

直接定義:如:String s1 = “myString”; 首先在String池中查找是否存在”myString”,如果沒有,在String池中創建”myString”,如果有,編譯期間直接指向該String池地址。 直接定義

使用關鍵字new:如:String s1 = new String(“myString”); 可以拆分成兩句理解,String buffer = “myString”;String s1 = new String(buffer);編譯期:首先檢查String池中有無myString,沒有就在String池中創建;運行期:為new的s1開辟堆內存空間,s1指向堆內存中”myString”。 使用關鍵字new

串聯生成:如:String s1 = “my” + “String”;或String s1 = “my”;String s2 = s1 + “String”; 這種方法比較復雜。如果是第一種常量拼接,編譯期間就能確定,s1指向String池;如果是第二種拼接,由于s1在編譯期間不能確定,所以s2在運行期間指向Java堆。

解釋:

接下來一個一個看:

1.

public static void m1() { String a = "a1"; String b = "a" + 1; System.out.println(a == b); // true}

首先在String池中創建”a1”,a指向String池中”a1”,然后在String池中創建”a”和”1”,b是前兩者的拼接,所以首先在String池中查找有無”a1”,有就直接指向,所以b也指向String池中的”a1”,在編譯期間就已經確定了內存地址,所以true。

2.

public static void m2() { String a = "ab"; String bb = "b"; String b = "a" + bb; System.out.println(a == b); // false}

a指向String池中的”ab”,bb指向String池中的”b”,b在編譯期間不能確定,所以在運行期間b會指向Java堆中的”ab”,所以false。

3.

public static void m3() { String a = "ab"; final String bb = "b"; String b = "a" + bb; System.out.println(a == b); // true}

a指向String池中”ab”,因為bb是final類型,所以在編譯期間會當做常量處理,bb在編譯期間等同于”b”,所以a和b都指向String池中”ab”,所以true。

4.

public static void m4() { String a = "ab"; final String bb = getBB(); String b = "a" + bb; System.out.println(a == b); // false}private static String getBB() { return "b";}

方法也是在運行期間才能確定,所以false。

5.

private static String a = "ab";public static void m5() { String s1 = "a"; String s2 = "b"; String s = s1 + s2; System.out.println(s == a); // false System.out.println(s.intern() == a); // true}

a指向String池中”ab”,s在編譯期間不確定,運行期間指向堆內存中”ab”,所以第一個是false;第二個用了intern()方法,返回String池中字符串,所以true。

6.

private static String ab = new String("ab");public static void m6() { String s1 = "a"; String s2 = "b"; String s = s1 + s2; System.out.println(s == ab); // false System.out.println(s.intern() == ab); // false System.out.println(s.intern() == ab.intern()); // true}

s1指向String池的”a”,s2指向String池的”b”,ab因為是new出來的,所以運行期間指向堆內存的”ab”,s編譯期間不能確定,在運行期間指向堆內存的”ab”,堆內存中每次都是創建新字符串,所以第一個是false;第二個由于用了intern()方法,返回String池中的”ab”,但ab指向的是堆內存的”ab”,所以第二個也是false;第三個由于都用了intern()方法,都是String池中的”ab”,所以true。

7.

private static void m7() { String s1 = "a"; String s2 = new String("a"); s2.intern(); System.out.println(s1 == s2); // false s2 = s2.intern(); System.out.println(s1 == s2); // true }

s2.intern();調用一下s2的intern()方法,首先在String池中尋找有無”a”,有直接返回引用,沒有的話先在String池中創建”a”,再返回引用。第一個輸出沒有賦值,所以false,第二個將s2指向String池中的”a”,所以true。

總結:

引用網上的不錯的總結:

原理1: 當使用任何方式來創建一個字符串對象s時,Java運行時(運行中JVM)會拿著這個X在String池中找是否存在內容相同的字符串對象,如果不存在,則在池中創建一個字符串s,否則,不在池中添加。

原理2: Java中,只要使用new關鍵字來創建對象,則一定會(在堆區或棧區)創建一個新的對象。

原理3: 使用直接指定或者使用純字符串串聯來創建String對象,則僅僅會檢查維護String池中的字符串,池中沒有就在池中創建一個,有則罷了!但絕不會在堆棧區再去創建該String對象。

原理4: 使用包含變量的表達式來創建String對象,則不僅會檢查維護String池,而且還會在堆棧區創建一個String對象。

Java常量池

在java中,不光String包含運用了常量池技術,java基本類型的包裝類的大部分都實現了常量池技術,這些類是Byte、Short、Integer、Long、Character、Boolean,另外兩種浮點數類型的包裝類則沒有實現。另外Byte、Short、Integer、Long、Character這5種整型的包裝類也只是在對應值小于等于127時才可使用常量池,也即常量池不負責創建和管理大于127的這些類的對象。常量池是為了方便快捷地創建某些對象而出現的,當需要一個對象時,就可以從池中取一個出來(如果池中沒有則創建一個),在需要重復創建相等變量時節省了很多時間。常量池其實也就是一個內存空間,不同于使用new關鍵字創建的對象所在的堆空間。


發表評論 共有條評論
用戶名: 密碼:
驗證碼: 匿名發表
主站蜘蛛池模板: 莲花县| 舒城县| 彭水| 云南省| 筠连县| 阳信县| 岳普湖县| 宁津县| 治县。| 洞口县| 上高县| 库伦旗| 建宁县| 辽源市| 宝兴县| 凉城县| 固阳县| 含山县| 龙里县| 平乡县| 绩溪县| 菏泽市| 古浪县| 阳城县| 内丘县| 会东县| 微山县| 东海县| 呼图壁县| 当阳市| 松溪县| 叶城县| 灵宝市| 漾濞| 无为县| 沂南县| 肥西县| 泰宁县| 镇沅| 大悟县| 闽侯县|