作者:臧圩人(zangweiren)
網址:http://zangweiren.javaeye.com
>>>轉載請註明出處!<<<
上一次我們已經一起回顧了面試題中常考的到底創建了幾個String對象的相關知識,這一次我們以幾個常見面試題爲引子,來回顧一下String對象相關的其它一些方面。
一、String類有length()方法嗎?數組有length()方法嗎?
String類當然有length()方法了,看看String類的源碼就知道了,這是這個方法的定義:
Java代碼
public int length() {
return count;
}
String的長度實際上就是它的屬性--char型數組value的長度。數組是沒有length()方法的,大家知道,在JAVA中,數組也被作爲對
象來處理,它的方法都繼承自Object類。數組有一個屬性length,這也是它唯一的屬性,對於所有類型的數組都是這樣。
二、一箇中文漢字能保存在一個char裏嗎?
請看下面的例子:
Java代碼
public class ChineseTest {
public static void main(String[] args) {
// 將一箇中文漢字賦值給一個char變量
char a = '中';
char b = '文';
char c = '測';
char d = '試';
char e = '成';
char f = '功';
System.out.print(a);
System.out.print(b);
System.out.print(c);
System.out.print(d);
System.out.print(e);
System.out.print(f);
}
}
編譯沒有報錯,運行結果:
1、中文測試成功
答案就不用說了。爲什麼一箇中文漢字可以保存在一個char變量裏呢?因爲在JAVA中,一個char是2個字節(byte),而一箇中文漢字是一個字符,也是2個字節。而英文字母都是一個字節的,因此它也能保存到一個byte裏,一箇中文漢字卻不能。請看:
Java代碼
public class ChineseTest {
public static void main(String[] args) {
// 將一個英文字母賦值給一個byte變量
byte a = 'a';
// 將一箇中文漢字賦值給一個byte變量時,編譯會報錯
// byte b = '中';
System.out.println("byte a = " + a);
// System.out.println("byte b = "+b);
}
}
運行結果:
1、byte a = 97
正如大家所看到的那樣,我們實際上是把字符'a'對應的ASCII碼值賦值給了byte型變量a。
讓我們回過頭來看看最初的例子,能不能將a、b、c、d、e、f拼接在一起一次輸出呢?讓我們試試看:
Java代碼
public class ChineseTest {
public static void main(String[] args) {
// 將一箇中文漢字賦值給一個char變量
char a = '中';
char b = '文';
char c = '測';
char d = '試';
char e = '成';
char f = '功';
System.out.print(a + b + c + d + e + f);
}
}
運行結果:
1、156035
這顯然不是我們想要的結果。只所以會這樣是因爲我們誤用了“+”運算符,當它被用於字符串和字符串之間,或者字符串和其他類型變量之間時,它產生的效果是
字符串的拼接;但當它被用於字符和字符之間時,效果等同於用於數字和數字之間,是一種算術運算。因此我們得到的“156035”是'中'、'文'、'測
'、'試'、'成'、'功'這六個漢字分別對應的數值算術相加後的結果。
三、字符串的反轉輸出。
這也是面試題中常考的一道。我們就以一個包含了全部26個英文字母,同時又具有完整含義的最短句子作爲例子來完成解答。先來看一下這個句子:
引用
A quick brown fox jumps over the lazy dog.(一隻輕巧的棕色狐狸從那條懶狗身上跳了過去。)
最常用的方式就是反向取出每個位置的字符,然後依次將它們輸出到控制檯:
Java代碼
public class StringReverse {
public static void main(String[] args) {
// 原始字符串
String s = "A quick brown fox jumps over the lazy dog.";
System.out.println("原始的字符串:" + s);
System.out.print("反轉後字符串:");
for (int i = s.length(); i > 0; i--) {
System.out.print(s.charAt(i - 1));
}
// 也可以轉換成數組後再反轉,不過有點多此一舉
char[] data = s.toCharArray();
System.out.println();
System.out.print("反轉後字符串:");
for (int i = data.length; i > 0; i--) {
System.out.print(data[i - 1]);
}
}
}
運行結果:
1、原始的字符串:A quick brown fox jumps over the lazy dog.
2、反轉後字符串:.god yzal eht revo spmuj xof nworb kciuq A
3、反轉後字符串:.god yzal eht revo spmuj xof nworb kciuq A
以上兩種方式雖然常用,但卻不是最簡單的方式,更簡單的是使用現有的方法:
Java代碼
public class StringReverse {
public static void main(String[] args) {
// 原始字符串
String s = "A quick brown fox jumps over the lazy dog.";
System.out.println("原始的字符串:" + s);
System.out.print("反轉後字符串:");
StringBuffer buff = new StringBuffer(s);
// java.lang.StringBuffer類的reverse()方法可以將字符串反轉
System.out.println(buff.reverse().toString());
}
}
運行結果:
1、原始的字符串:A quick brown fox jumps over the lazy dog.
2、反轉後字符串:.god yzal eht revo spmuj xof nworb kciuq A
四、按字節截取含有中文漢字的字符串。
要求實現一個按字節截取字符串的方法,比如對於字符串"我ZWR愛JAVA",截取它的前四位字節應該是"我ZW",而不是"我ZWR",同時要保證不會出現截取了半個漢字的情況。
英文字母和中文漢字在不同的編碼格式下,所佔用的字節數也是不同的,我們可以通過下面的例子來看看在一些常見的編碼格式下,一個英文字母和一箇中文漢字分別佔用多少字節。
Java代碼
import java.io.UnsupportedEncodingException;
public class EncodeTest {
/**
* 打印字符串在指定編碼下的字節數和編碼名稱到控制檯
*
* @param s
* 字符串
* @param encodingName
* 編碼格式
*/
public static void printByteLength(String s, String encodingName) {
System.out.print("字節數:");
try {
System.out.print(s.getBytes(encodingName).length);
} catch (UnsupportedEncodingException e) {
e.printStackTrace();
}
System.out.println(";編碼:" + encodingName);
}
public static void main(String[] args) {
String en = "A";
String ch = "人";
// 計算一個英文字母在各種編碼下的字節數
System.out.println("英文字母:" + en);
EncodeTest.printByteLength(en, "GB2312");
EncodeTest.printByteLength(en, "GBK");
EncodeTest.printByteLength(en, "GB18030");
EncodeTest.printByteLength(en, "ISO-8859-1");
EncodeTest.printByteLength(en, "UTF-8");
EncodeTest.printByteLength(en, "UTF-16");
EncodeTest.printByteLength(en, "UTF-16BE");
EncodeTest.printByteLength(en, "UTF-16LE");
System.out.println();
// 計算一箇中文漢字在各種編碼下的字節數
System.out.println("中文漢字:" + ch);
EncodeTest.printByteLength(ch, "GB2312");
EncodeTest.printByteLength(ch, "GBK");
EncodeTest.printByteLength(ch, "GB18030");
EncodeTest.printByteLength(ch, "ISO-8859-1");
EncodeTest.printByteLength(ch, "UTF-8");
EncodeTest.printByteLength(ch, "UTF-16");
EncodeTest.printByteLength(ch, "UTF-16BE");
EncodeTest.printByteLength(ch, "UTF-16LE");
}
}
運行結果如下:
1、英文字母:A
2、字節數:1;編碼:GB2312
3、字節數:1;編碼:GBK
4、字節數:1;編碼:GB18030
5、字節數:1;編碼:ISO-8859-1
6、字節數:1;編碼:UTF-8
7、字節數:4;編碼:UTF-16
8、字節數:2;編碼:UTF-16BE
9、字節數:2;編碼:UTF-16LE
10、
11、中文漢字:人
12、字節數:2;編碼:GB2312
13、字節數:2;編碼:GBK
14、字節數:2;編碼:GB18030
15、字節數:1;編碼:ISO-8859-1
16、字節數:3;編碼:UTF-8
17、字節數:4;編碼:UTF-16
18、字節數:2;編碼:UTF-16BE
19、字節數:2;編碼:UTF-16LE
UTF-16BE和UTF-16LE是UNICODE編碼家族的兩個成員。UNICODE標準定義了UTF-8、UTF-16、UTF-32三種編碼格
式,共有UTF-8、UTF-16、UTF-16BE、UTF-16LE、UTF-32、UTF-32BE、UTF-32LE七種編碼方案。JAVA所採
用的編碼方案是UTF-16BE。從上例的運行結果中我們可以看出,GB2312、GBK、GB18030三種編碼格式都可以滿足題目的要求。下面我們就
以GBK編碼爲例來進行解答。
我們不能直接使用String類的substring(int beginIndex, int
endIndex)方法,因爲它是按字符截取的。'我'和'Z'都被作爲一個字符來看待,length都是1。實際上我們只要能區分開中文漢字和英文字
母,這個問題就迎刃而解了,而它們的區別就是,中文漢字是兩個字節,英文字母是一個字節。
Java代碼
import java.io.UnsupportedEncodingException;
public class CutString {
/**
* 判斷是否是一箇中文漢字
*
* @param c
* 字符
* @return true表示是中文漢字,false表示是英文字母
* @throws UnsupportedEncodingException
* 使用了JAVA不支持的編碼格式
*/
public static boolean isChineseChar(char c)
throws UnsupportedEncodingException {
// 如果字節數大於1,是漢字
// 以這種方式區別英文字母和中文漢字並不是十分嚴謹,但在這個題目中,這樣判斷已經足夠了
return String.valueOf(c).getBytes("GBK").length > 1;
}
/**
* 按字節截取字符串
*
* @param orignal
* 原始字符串
* @param count
* 截取位數
* @return 截取後的字符串
* @throws UnsupportedEncodingException
* 使用了JAVA不支持的編碼格式
*/
public static String substring(String orignal, int count)
throws UnsupportedEncodingException {
// 原始字符不爲null,也不是空字符串
if (orignal != null && !"".equals(orignal)) {
// 將原始字符串轉換爲GBK編碼格式
orignal = new String(orignal.getBytes(), "GBK");
// 要截取的字節數大於0,且小於原始字符串的字節數
if (count > 0 && count < orignal.getBytes("GBK").length) {
StringBuffer buff = new StringBuffer();
char c;
for (int i = 0; i < count; i++) {
c = orignal.charAt(i);
buff.append(c);
if (CutString.isChineseChar(c)) {
// 遇到中文漢字,截取字節總數減1
--count;
}
}
return buff.toString();
}
}
return orignal;
}
public static void main(String[] args) {
// 原始字符串
String s = "我ZWR愛JAVA";
System.out.println("原始字符串:" + s);
try {
System.out.println("截取前1位:" + CutString.substring(s, 1));
System.out.println("截取前2位:" + CutString.substring(s, 2));
System.out.println("截取前4位:" + CutString.substring(s, 4));
System.out.println("截取前6位:" + CutString.substring(s, 6));
} catch (UnsupportedEncodingException e) {
e.printStackTrace();
}
}
}
運行結果:
1、原始字符串:我ZWR愛JAVA
2、截取前1位:我
3、截取前2位:我
4、截取前4位:我ZW
5、截取前6位:我ZWR愛
JAVA面試題解惑系列(六)——字符串(String)雜談
發表評論
所有評論
還沒有人評論,想成為第一個評論的人麼? 請在上方評論欄輸入並且點擊發布.