String類常見問題

轉自: 小黃鴨編程社區
1.判定定義爲String類型的st1和st2是否相等,爲什麼?

public static void stringTest1(){
        String st1 = "abc";
        String st2 = "abc";
        System.out.println(st1 == st2);
        System.out.println(st1.equals(st2));
}

輸出結果:
第一行:true
第二行:true
分析:
先看第一個打印語句,在Java中==這個符號是比較運算符,它可以基本數據類型和引用數據類型是否相等,如果是基本數據類型,==比較的是值是否相等,如果是引用數據類型,==比較的是兩個對象的內存地址是否相等。

字符串不屬於8中基本數據類型,字符串對象屬於引用數據類型,在上面把“abc”同時賦值給了st1和st2兩個字符串對象,指向的都是同一個地址,所以第一個打印語句中的==比較輸出結果是 true

然後我們看第二個打印語句中的equals的比較,我們知道,equals是Object這個父類的方法,在String類中重寫了這個equals方法。

在JDK API 1.6文檔中找到String類下的equals方法,點擊進去可以看大這麼一句話“將此字符串與指定的對象比較。當且僅當該參數不爲null,並且是與此對象表示相同字符序列的String 對象時,結果才爲 true。” 注意這個相同字符序列,在後面介紹的比較兩個數組,列表,字典是否相等,都是這個邏輯去寫代碼實現。

由於st1和st2的值都是“abc”,兩者指向同一個對象,當前字符序列相同,所以第二行打印結果也爲true。

下面我們來畫一個內存圖來表示上面的代碼,看起來更加有說服力。
在這裏插入圖片描述
內存過程大致如下:
1)運行先編譯,然後當前類Demo2_String.class文件加載進入內存的方法區
2)第二步,main方法壓入棧內存
3)常量池創建一個“abc”對象,產生一個內存地址
4)然後把“abc”內存地址賦值給main方法裏的成員變量st1,這個時候st1根據內存地址,指向了常量池中的“abc”。
5)前面一篇提到,常量池有這個特點,如果發現已經存在,就不在創建重複的對象
6)運行到代碼 Stringst2 =”abc”, 由於常量池存在“abc”,所以不會再創建,直接把“abc”內存地址賦值給了st2
7)最後st1和st2都指向了內存中同一個地址,所以兩者是完全相同的。

2.下面這句話在內存中創建了幾個對象

String st1 = new String(“abc”);

答案是:在內存中創建兩個對象,一個在堆內存,一個在常量池,堆內存對象是常量池對象的一個拷貝副本。

分析:
我們下面直接來一個內存圖。

在這裏插入圖片描述
當我們看到了new這個關鍵字,就要想到,new出來的對象都是存儲在堆內存。然後我們來解釋堆中對象爲什麼是常量池的對象的拷貝副本。

“abc”屬於字符串,字符串屬於常量,所以應該在常量池中創建,所以第一個創建的對象就是在常量池裏的“abc”。

第二個對象在堆內存爲啥是一個拷貝的副本呢,這個就需要在JDK API 1.6找到String(String original)這個構造方法的註釋:初始化一個新創建的 String 對象,使其表示一個與參數相同的字符序列;換句話說,新創建的字符串是該參數字符串的副本。
所以,答案就出來了,兩個對象。

3、判定以下定義爲String類型的st1和st2是否相等

public static void stringTest2(){
    String st1 = new String("abc");
    String st2 = "abc";
    System.out.println(st1 == st2);
    System.out.println(st1.equals(st2));
}

答案:false 和 true
由於有前面兩道提內存分析的經驗和理論,所以,我能快速得出上面的答案。
==比較的st1和st2對象的內存地址,由於st1指向的是堆內存的地址,st2看到“abc”已經在常量池存在,就不會再新建,所以st2指向了常量池的內存地址,所以 ,==判斷結果輸出false,兩者不相等。

第二個equals比較,比較是兩個字符串序列是否相等,由於就一個“abc”,所以完全相等。
內存圖如下
在這裏插入圖片描述
4. 判定以下定義爲String類型的st1和st2是否相等

public static void stringTest3(){
    String st1 = "a" + "b" + "c";
    String st2 = "abc";
    System.out.println(st1 == st2);
    System.out.println(st1.equals(st2));
}

答案是:true 和 true
分析:
“a”,”b”,”c”三個本來就是字符串常量,進行+符號拼接之後變成了“abc”,“abc”本身就是字符串常量(Java中有常量優化機制),所以常量池立馬會創建一個“abc”的字符串常量對象,在進行st2=”abc”,這個時候,常量池存在“abc”,所以不再創建。所以,不管比較內存地址還是比較字符串序列,都相等。

5、判斷以下st2和st3是否相等

public static void stringTest5(){
    String st1 = "ab";
    String st2 = "abc";
    String st3 = st1 + "c";
    System.out.println(st2 == st3);
    System.out.println(st2.equals(st3));
}

答案:false 和 true
分析:

上面的答案第一個是false,第二個是true,第二個是true我們很好理解,因爲比較一個是“abc”,另外一個是拼接得到的“abc”,所以equals比較,這個是輸出true,我們很好理解。

那麼第一個判斷爲什麼是false,我們很疑惑。同樣,下面我們用API的註釋說明和內存圖來解釋這個爲什麼不相等。
首先,打開JDK API 1.6中String的介紹,找到下面圖片這句話。
在這裏插入圖片描述
關鍵點就在紅圈這句話,我們知道任何數據和字符串進行加號(+)運算,最終得到是一個拼接的新的字符串。
上面註釋說明了這個拼接的原理是由StringBuilder或者StringBuffer類和裏面的append方法實現拼接,然後調用 toString() 把拼接的對象轉換成字符串對象,最後把得到字符串對象的地址賦值給變量。
結合這個理解,我們下面畫一個內存圖來分析
在這裏插入圖片描述
大致內存過程
1)常量池創建“ab”對象,並賦值給st1,所以st1指向了“ab”
2)常量池創建“abc”對象,並賦值給st2,所以st2指向了“abc”
3)由於這裏走的+的拼接方法,所以第三步是使用StringBuffer類的append方法,得到了“abc”,這個時候內存0x0011表示的是一個StringBuffer對象,注意不是String對象。4)調用了Object的toString方法把StringBuffer對象裝換成了String對象。
5)把String對象(0x0022)賦值給st3

所以,st3和st2進行==判斷結果是不相等,因爲兩個對象內存地址不同。

總結:

主要比較是 == 和equals 的比較,通俗的理解 ==是比較兩個字符串的引用地址是否一致,equals 是判斷兩個字符串是否長的一樣

常用方法總結

獲取功能的方法
public int length () :返回此字符串的長度。
public String concat (String str) :將指定的字符串連接到該字符串的末尾。
public char charAt (int index) :返回指定索引處的 char值。
public int indexOf (String str) :返回指定子字符串第一次出現在該字符串內的索引。
public String substring (int beginIndex) :返回一個子字符串,從beginIndex開始截取字符串到字符串結尾。
public String substring (int beginIndex, int endIndex) :返回一個子字符串,從beginIndex到endIndex截取字符串。含beginIndex,不含endIndex。
public int compareTo(String anotherString)//該方法是對字符串內容按字典順序進行大小比較,通過返回的整數值指明當前字符串與參數字符串的大小關係。若當前對象比參數大則返回正整數,反之返回負整數,相等返回0。
public int compareToIgnore(String anotherString)//與compareTo方法相似,但忽略大小寫。
public boolean equals(Object anotherObject)//比較當前字符串和參數字符串,在兩個字符串相等的時候返回true,否則返回false。
public boolean equalsIgnoreCase(String anotherString)//與equals方法相似,但忽略大小寫。
public String concat(String str)//將參數中的字符串str連接到當前字符串的後面,效果等價於"+"。
public int indexOf(int ch/String str)//用於查找當前字符串中字符或子串,返回字符或子串在當前字符串中從左邊起首次出現的位置,若沒有出現則返回-1。
)public int indexOf(int ch/String str, int fromIndex)//改方法與第一種類似,區別在於該方法從fromIndex位置向後查找。
public int lastIndexOf(int ch/String str)//該方法與第一種類似,區別在於該方法從字符串的末尾位置向前查找。
public int lastIndexOf(int ch/String str, int fromIndex)//該方法與第二種方法類似,區別於該方法從fromIndex位置向前查找。
public String toLowerCase()//返回將當前字符串中所有字符轉換成小寫後的新串
public String toUpperCase()//返回將當前字符串中所有字符轉換成大寫後的新串
public String replace(char oldChar, char newChar)//用字符newChar替換當前字符串中所有的oldChar字符,並返回一個新的字符串。
public String replaceFirst(String regex, String replacement)//該方法用字符replacement的內容替換當前字符串中遇到的第一個和字符串regex相匹配的子串,應將新的字符串返回。
public String replaceAll(String regex, String replacement)//該方法用字符replacement的內容替換當前字符串中遇到的所有和字符串regex相匹配的子串,應將新的字符串返回。
String trim()//截去字符串兩端的空格,但對於中間的空格不處理。
boolean statWith(String prefix)或boolean endWith(String suffix)//用來比較當前字符串的起始字符或子字符串prefix和終止字符或子字符串suffix是否和當前字符串相同,重載方法中同時還可以指定比較的開始位置offset。
regionMatches(boolean b, int firstStart, String other, int otherStart, int length)//從當前字符串的firstStart位置開始比較,取長度爲length的一個子字符串,other字符串從otherStart位置開始,指定另外一個長度爲length的字符串,兩字符串比較,當b爲true時字符串不區分大小寫。
contains(String str)//判斷參數s是否被包含在字符串中,並返回一個布爾類型的值。

轉換功能的方法
public char[] toCharArray () :將此字符串轉換爲新的字符數組。
public byte[] getBytes () :使用平臺的默認字符集將該 String編碼轉換爲新的字節數組。
public String replace (CharSequence target, CharSequence replacement) :將與target匹配的字符串使用replacement字符串替換。
分割功能的方法
public String[] split(String regex) :將此字符串按照給定的regex(規則)拆分爲字符串數組。
1、字符串轉換爲基本類型
java.lang包中有Byte、Short、Integer、Float、Double類的調用方法:
1)public static byte parseByte(String s)
2)public static short parseShort(String s)
3)public static short parseInt(String s)
4)public static long parseLong(String s)
5)public static float parseFloat(String s)
6)public static double parseDouble(String s)
2、基本類型轉換爲字符串類型
String類中提供了String valueOf()放法,用作基本類型轉換爲字符串類型。
1)static String valueOf(char data[])
2)static String valueOf(char data[], int offset, int count)
3)static String valueOf(boolean b)
4)static String valueOf(char c)
5)static String valueOf(int i)
6)static String valueOf(long l)
7)static String valueOf(float f)
8)static String valueOf(double d)

發表評論
所有評論
還沒有人評論,想成為第一個評論的人麼? 請在上方評論欄輸入並且點擊發布.
相關文章