天天看點

java中了解字元串常量池的幾個例子

package juc.base;

/*1.字面量方式定義(例如String a = "abc")會檢查字元串常量池中是否已經有這個字元串的引用,
如果有,就直接傳回引用,沒有,在堆中建立字元串對象對象再傳回引用,這種情況下隻會建立
一個字元串對象,并且這個字元串對象的引用已經存儲在字元串常量池*/
/*2.new關鍵字定義(例如String a = new ("abc")),會在堆中建立一個字元串對象@111111(假如事先不存在),
并把引用存儲在字元串常量池,并會在堆中再建立一個字元串變量@222222,注意,@111111和@222222雖然内容一樣
但是他們時兩個不同對象(記憶體位址不同),并且傳回的時@222222這個引用*/
/*3.toString()方式(例如 String a = new StringBuilder("a").append("bc").toString()),
會建立4個對象,"a","bc"在堆中建立,并在字元串常量池中存儲引用(假如事先在字元串常量池中不存在),一個StringBuilder對象在
堆中建立,一個"abc字"符串對象在堆中建立,特别注意,***此時"abc"對象的索引并不會在存儲在字元串常量池中***
此時傳回的時堆中的"abc"這個對象*/
/*4.intern()方法會先判斷字元串常量池中是否有改字元串的引用,如果有,直接傳回該引用,
如果沒有,則存儲該對象的引用,最後傳回該對象的引用,注意,***字元串常量池隻存儲引用***
可以使用一個HashSet結構來了解,HashSet strSet = new HashSet<String>();*/
public class StringDemo {
    public static void main(String[] args) {
//        情況1:傳回true
        String a = "張三";
        System.out.println(a==a.intern());

//        情況1:傳回false
        String b = new String("李四");
        System.out.println(b == b.intern());

//        情況3:傳回true
        String c = new StringBuilder("王").append("五").toString();
        System.out.println(c==c.intern());

//        情況4:傳回false
        /*@@@@@@******這是一個大坑,看似和例子3幾乎一模一樣,但是需要注意的時,"java"這個字元串,
        在JVM加載後就已經被存儲到了字元串常量池,是以在執行d.intern()時實際不會
        把原先堆中的那個"java"字元串對象的以用存儲到字元串常量池中,是以2者并不相等*/
        String d = new StringBuilder("ja").append("va").toString();
        System.out.println(d==d.intern());
    }
}