String類的使用2

/*
String:字符串,使用一對""引發來表示。
1.String聲明爲final的,不可被繼承
2.String實現了Serializable接口:表示字符串是支持序列化的。
實現了Comparable接口:表示String能夠比較大小
3.String內部定義了final char[] value用於存儲字符串數據
4.String:表明不可變的字符序列。簡稱:不可變性。
體現:1.當對字符串從新賦值時,須要重寫指定內存區域賦值,不能使用原有的value進行賦值。
2. 當對現有的字符串進行鏈接操做時,也須要從新指定內存區域賦值,不能使用原有的value進行賦值。
3. 當調用String的replace()方法修改指定字符或字符串時,也須要從新指定內存區域賦值,不能使用原有的value進行賦值。
5.經過字面量的方式(區別於new)給一個字符串賦值,此時的字符串值聲明在字符串常量池中。
6.字符串常量池中是不會存儲相同內容的字符串的。
*/
 @Test
    public void test1(){
        String s1 = "abc";//字面量的定義方式
        String s2 = "abc";
        s1 = "hello";

        System.out.println(s1 == s2);//比較s1和s2的地址值

        System.out.println(s1);//hello
        System.out.println(s2);//abc

        System.out.println("*****************");

        String s3 = "abc";
        s3 += "def";
        System.out.println(s3);//abcdef
        System.out.println(s2);

        System.out.println("*****************");

        String s4 = "abc";
        String s5 = s4.replace('a', 'm');
        System.out.println(s4);//abc
        System.out.println(s5);//mbc

    }

 

 

 
/*
String的實例化方式:
方式一:經過字面量定義的方式
方式二:經過new + 構造器的方式

面試題:String s = new String("abc");方式建立對象,在內存中建立了幾個對象?
兩個:一個是堆空間中new結構,另外一個是char[]對應的常量池中的數據:"abc"

*/
 
@Test
    public void test2(){
        //經過字面量定義的方式:此時的s1和s2的數據javaEE聲明在方法區中的字符串常量池中。
        String s1 = "javaEE";
        String s2 = "javaEE";
        //經過new + 構造器的方式:此時的s3和s4保存的地址值,是數據在堆空間中開闢空間之後對應的地址值。
        String s3 = new String("javaEE");
        String s4 = new String("javaEE");

        System.out.println(s1 == s2);//true
        System.out.println(s1 == s3);//false
        System.out.println(s1 == s4);//false
        System.out.println(s3 == s4);//false

        System.out.println("***********************");
        Person p1 = new Person("Tom",12);
        Person p2 = new Person("Tom",12);

        System.out.println(p1.name.equals(p2.name));//true
        System.out.println(p1.name == p2.name);//true

        p1.name = "Jerry";
        System.out.println(p2.name);//Tom
    }
 

 

 

 
/*
String 與 byte[]之間的轉換
編碼:String --> byte[]:調用String的getBytes()
解碼:byte[] --> String:調用String的構造器

編碼:字符串 -->字節 (看得懂 --->看不懂的二進制數據)
解碼:編碼的逆過程,字節 --> 字符串 (看不懂的二進制數據 ---> 看得懂)

說明:解碼時,要求解碼使用的字符集必須與編碼時使用的字符集一致,不然會出現亂碼。
*/
 
@Test
    public void test3() throws UnsupportedEncodingException {
        String str1 = "abc123中國";
        byte[] bytes = str1.getBytes();//使用默認的字符集,進行編碼。
        System.out.println(Arrays.toString(bytes));

        byte[] gbks = str1.getBytes("gbk");//使用gbk字符集進行編碼。
        System.out.println(Arrays.toString(gbks));

        System.out.println("******************");

        String str2 = new String(bytes);//使用默認的字符集,進行解碼。
        System.out.println(str2);

        String str3 = new String(gbks);
        System.out.println(str3);//出現亂碼。緣由:編碼集和解碼集不一致!


        String str4 = new String(gbks, "gbk");
        System.out.println(str4);//沒有出現亂碼。緣由:編碼集和解碼集一致!


    }
 

 

 

 
/*
結論:
1.常量與常量的拼接結果在常量池。且常量池中不會存在相同內容的常量。
2.只要其中有一個是變量,結果就在堆中。
3.若是拼接的結果調用intern()方法,返回值就在常量池中
*/
@Test
    public void test4(){
        String s1 = "javaEEhadoop";
        String s2 = "javaEE";
        String s3 = s2 + "hadoop";
        System.out.println(s1 == s3);//false

        final String s4 = "javaEE";//s4:常量
        String s5 = s4 + "hadoop";
        System.out.println(s1 == s5);//true

    }

    @Test
    public void test3(){
        String s1 = "javaEE";
        String s2 = "hadoop";

        String s3 = "javaEEhadoop";
        String s4 = "javaEE" + "hadoop";
        String s5 = s1 + "hadoop";
        String s6 = "javaEE" + s2;
        String s7 = s1 + s2;

        System.out.println(s3 == s4);//true
        System.out.println(s3 == s5);//false
        System.out.println(s3 == s6);//false
        System.out.println(s3 == s7);//false
        System.out.println(s5 == s6);//false
        System.out.println(s5 == s7);//false
        System.out.println(s6 == s7);//false

        String s8 = s6.intern();//返回值獲得的s8使用的常量值中已經存在的「javaEEhadoop」
        System.out.println(s3 == s8);//true


    }
相關文章
相關標籤/搜索