String是最常使用的Java類之一簿姨,整理的了一些重要的String知識分享給大家羡榴。
作為一個Java新手程序員,對String進(jìn)行更深入的了解很有必要。如果你是有幾年Java開發(fā)經(jīng)驗豁护,可以根據(jù)目錄選擇性的閱讀以下內(nèi)容。
1专钉、什么是String图柏,它是什么數(shù)據(jù)類型?
String是定義在 java.lang 包下的一個類报破。它不是基本數(shù)據(jù)類型悠就。
String是不可變的,JVM使用字符串池來存儲所有的字符串對象充易。
2梗脾、創(chuàng)建String對象的不同方式有哪些?
-
和使用其他類一樣通過new關(guān)鍵字來創(chuàng)建盹靴。
使用這種方式時炸茧,JVM創(chuàng)建字符串對象但不存儲于字符串池瑞妇。我們可以調(diào)用intern()方法將該字符串對象存儲在字符串池,如果字符串池已經(jīng)有了同樣值的字符串梭冠,則返回引用辕狰。
使用雙引號直接創(chuàng)建。
使用這種方式時控漠,JVM去字符串池找有沒有值相等字符串蔓倍,如果有,則返回找到的字符串引用盐捷。否則創(chuàng)建一個新的字符串對象并存儲在字符串池偶翅。
String str = new String("abc");
String str1 = "abc";
3、寫一個方法來判斷一個String是否是回文(順讀和倒讀都一樣的詞)碉渡?
回文就是正反都一樣的詞聚谁,如果需要判斷是否是回文,只需要比較正反是否相等即可滞诺。String類并沒有提供反轉(zhuǎn)方法供我們使用垦巴,但StringBuffer和StringBuilder有reverse方法。
private static boolean isPalindrome(String str) {
if (str == null)
return false;
StringBuilder strBuilder = new StringBuilder(str);
strBuilder.reverse();
return strBuilder.toString().equals(str);
}
假設(shè)面試官讓你不使用任何其他類來實現(xiàn)的話铭段,我們只需要首尾一一對比就知道是不是回文了骤宣。
private static boolean isPalindromeString(String str) {
if (str == null)
return false;
int length = str.length();
System.out.println(length / 2);
for (int i = 0; i < length / 2; i++) {
if (str.charAt(i) != str.charAt(length - i - 1))
return false;
}
return true;
}
4、如何讓一個字符串變成小寫或大寫形式序愚?
使用toUpperCase 和 toLowerCase 方法讓一個字符串變?yōu)?大寫或小寫憔披。
5、如何比較兩個字符串爸吮?
String內(nèi)部實現(xiàn)了Comparable接口芬膝,有兩個比較方法:compareTo(String anotherString) 和compareToIgnoreCase(String str)。
-
compareTo(String anotherString)
與傳入的anotherString字符串進(jìn)行比較形娇,如果小于傳入的字符串返回負(fù)數(shù)锰霜,如果大于則返回證書。當(dāng)兩個字符串值相等時桐早,返回0.此時eqauls方法會返回true癣缅。
-
equalsIgnoreCase(String str)
該方法與compareTo方法類似,區(qū)別只是內(nèi)部利用了Character.toUpperCase等方法進(jìn)行了大小寫轉(zhuǎn)換后進(jìn)行比較哄酝。
6友存、如何將String轉(zhuǎn)換為char,反過來呢?
這是一個誤導(dǎo)題陶衅,String是一系列字符屡立,所有我們沒法轉(zhuǎn)換成一個單一的char,但可以調(diào)用toCharArray() 方法將字符串轉(zhuǎn)成字符數(shù)組搀军。
String str = "Java interview";
//string to char array
char[] chars = str.toCharArray();
System.out.println(chars.length);
7膨俐、如何將String轉(zhuǎn)換為byte array,反過來呢勇皇?
使用String的getBytes()方法將String轉(zhuǎn)成byte數(shù)組,使用String的構(gòu)造方法 new String(byte[] arr) 將byte數(shù)據(jù)轉(zhuǎn)為String焚刺。
public class StringToByteArray {
public static void main(String[] args) {
String str = "PANKAJ";
byte[] byteArr = str.getBytes();
// print the byte[] elements
System.out.println("String to byte array: " + Arrays.toString(byteArr));
}
}
public class ByteArrayToString {
public static void main(String[] args) {
byte[] byteArray = { 'P', 'A', 'N', 'K', 'A', 'J' };
byte[] byteArray1 = { 80, 65, 78, 75, 65, 74 };
String str = new String(byteArray);
String str1 = new String(byteArray1);
System.out.println(str);
System.out.println(str1);
}
}
<div id="question8"></div>
8敛摘、淺談一下String, StringBuffer,StringBuilder的區(qū)別檩坚?
String是不可變類,每當(dāng)我們對String進(jìn)行操作的時候诅福,總是會創(chuàng)建新的字符串匾委。操作String很耗資源,所以Java提供了兩個工具類來操作String - StringBuffer和StringBuilder。
StringBuffer和StringBuilder是可變類氓润,StringBuffer是線程安全的赂乐,StringBuilder則不是線程安全的。所以在多線程對同一個字符串操作的時候咖气,我們應(yīng)該選擇用StringBuffer挨措。由于不需要處理多線程的情況,StringBuilder的效率比StringBuffer高崩溪。
9浅役、String是不可變的有什么好處?
String是不可變類有以下幾個優(yōu)點
- 由于String是不可變類伶唯,所以在多線程中使用是安全的觉既,我們不需要做任何其他同步操作。
- String是不可變的乳幸,它的值也不能被改變瞪讼,所以用來存儲數(shù)據(jù)密碼很安全。
- 因為java字符串是不可變的粹断,可以在java運行時節(jié)省大量java堆空間符欠。因為不同的字符串變量可以引用池中的相同的字符串。如果字符串是可變得話瓶埋,任何一個變量的值改變希柿,就會反射到其他變量,那字符串池也就沒有任何意義了养筒。
10狡汉、如何分割一個String?
-
public String[] split(String regex):
根據(jù)傳入的正則字符串進(jìn)行分割闽颇,注意盾戴,如果最后一位剛好有傳入的字符,返回數(shù)組最后一位不會有空字符串兵多。
String s = "abcaada";
System.out.println(Arrays.toString(s.split("a")));
//以上代碼輸出為 [, bc, , d].
-
public String[] split(String regex, int limit):
限制分割結(jié)果數(shù)組中有幾個字符串尖啡。傳入2,則結(jié)果分割后數(shù)組長度為2橄仆。
String s = "Y,Kunming,Yunnan";
String[] data = s.split(",", 2);
System.out.println("Name = "+data[0]); //Y
System.out.println("Address = "+data[1]); //Kunming,Yunnan
實際上第一個方法調(diào)用了第二個方法,只不過不限制返回的數(shù)組長度了衅斩。
public String[] split(String regex) {
return split(regex, 0);
}
11盆顾、如何判斷兩個String是否相等?
有兩種方式判斷字符串是否相等畏梆,使用"=="或者使用equals方法您宪。當(dāng)使用"=="操作符時,不僅比較字符串的值奠涌,還會比較引用的內(nèi)存地址宪巨。大多數(shù)情況下,我們只需要判斷值是否相等溜畅,此時用equals方法比較即可捏卓。
還有一個equalsIgnoreCase可以用來忽略大小寫進(jìn)行比較。
String s1 = "abc";
String s2 = "abc";
String s3= new String("abc");
System.out.println("s1 == s2 ? "+(s1==s2)); //true
System.out.println("s1 == s3 ? "+(s1==s3)); //false
System.out.println("s1 equals s3 ? "+(s1.equals(s3))); //true
12慈格、什么是字符串池怠晴?
顧名思義,字符串常量池就是用來存儲字符串的浴捆。它存在于Java 堆內(nèi)存蒜田。
下圖解釋了字符串池在java堆空間如何存在以及當(dāng)我們使用不同方式創(chuàng)建字符串時的情況。
以下是上圖的一個編程例子
public class StringPool {
public static void main(String[] args) {
String s1 = "Cat";
String s2 = "Cat";
String s3 = new String("Cat");
System.out.println("s1 == s2 :"+(s1==s2));
System.out.println("s1 == s3 :"+(s1==s3));
}
}
運行以上代碼选泻,輸出如下:
s1 == s2 :true
s1 == s3 :false
一些java題中物邑,可能會問一段代碼中有幾個字符串被創(chuàng)建,例如:
String str = new String("Cat");
上面一行代碼將會創(chuàng)建1或2個字符串滔金。如果在字符串常量池中已經(jīng)有一個字符串“Cat”色解,那么就智慧創(chuàng)建一個“Cat”字符串。如果字符串常量池中沒有“Cat”餐茵,那么首先會在字符串池中創(chuàng)建科阎,然后才在堆內(nèi)存中創(chuàng)建,這種情況就會創(chuàng)建2個對象了忿族。
13锣笨、String的intern()方法
當(dāng)intern()方法被調(diào)用,如果字符串池中含有一個字符串和當(dāng)前調(diào)用方法的字符串eqauls相等道批,那么就會返回池中的字符串错英。如果池中沒有的話,則首先將當(dāng)前字符串加入到池中隆豹,然后返回引用椭岩。
14、String是線程安全的嗎?
String是不可變類判哥,一旦創(chuàng)建了String對象献雅,我們就無法改變它的值。因此塌计,它是線程安全的挺身,可以安全地用于多線程環(huán)境中。
15锌仅、為什么我們在使用HashMap的時候總是用String做key章钾?
因為字符串是不可變的,當(dāng)創(chuàng)建字符串時热芹,它的它的hashcode被緩存下來贱傀,不需要再次計算。因為HashMap內(nèi)部實現(xiàn)是通過key的hashcode來確定value的存儲位置剿吻,所以相比于其他對象更快窍箍。這也是為什么我們平時都使用String作為HashMap對象串纺。
16丽旅、String編程題
1、下面的代碼輸入什么
String s1 = new String("abc");
String s2 = new String("abc");
System.out.println(s1 == s2);
輸入false
2纺棺、下面的代碼輸入什么
String s1 = "abc";
StringBuffer s2 = new StringBuffer(s1);
System.out.println(s1.equals(s2));
輸入false榄笙,因為s2不是String類型,String的equals方法進(jìn)行了類型判斷祷蝌。
3茅撞、下面的代碼輸入什么
String s1 = "abc";
String s2 = new String("abc");
s2.intern();
System.out.println(s1 ==s2);
輸出false,intern()方法將返回從字符串池中的字符串對象的引用巨朦,但因為我們沒有分配到S2米丘,S2沒有變化,如果該第三行代碼為s2 =
s2.intern()糊啡,則輸入true拄查。
4、下面的代碼將創(chuàng)建幾個字符串對象棚蓄。
String s1 = new String("Hello");
String s2 = new String("Hello");
答案是3個對象.
第一堕扶,行1 字符串池中的“hello”對象。
第二梭依,行1稍算,在堆內(nèi)存中帶有值“hello”的新字符串。
第三役拴,行2糊探,在堆內(nèi)存中帶有“hello”的新字符串。這里“hello”字符串池中的字符串被重用。
題外話
希望上面的這些知識能夠幫助到你侧到,如果有哪些不足請留言告知勃教!