Java基礎(chǔ)總結(jié)——String

1 String是不可變類(lèi)

這句話其實(shí)大家都很熟悉了,那么具體什么是不可變類(lèi)呢盹兢?一般認(rèn)為:當(dāng)對(duì)象一旦創(chuàng)建完成后锉试,在正常情況下玻募,對(duì)象的狀態(tài)不會(huì)因外界的改變而改變(對(duì)象的狀態(tài)是指對(duì)象的屬性蚓再,包括屬性的類(lèi)型及屬性值)滑肉。

例如:

String s = "Google";
System.out.println("s = " + s);

s = "Runoob";
System.out.println("s = " + s);

輸出結(jié)果為:

s = Google
s = Runoob

從結(jié)果上看是改變了,但為什么門(mén)說(shuō)String對(duì)象是不可變的呢摘仅?
原因在于實(shí)例中的 s只是指向堆內(nèi)存中的引用靶庙,存儲(chǔ)的是對(duì)象在堆中的地址,而非對(duì)象本身实檀,s本身存儲(chǔ)在棧內(nèi)存中

當(dāng)執(zhí)行 s = "Runoob"; 創(chuàng)建了一個(gè)新的對(duì)象 "Runoob"惶洲,而原來(lái)的 "Google" 還存在于內(nèi)存中。

那么為什么String類(lèi)具有不可變性呢膳犹,顯然恬吕,既然不可變說(shuō)明String類(lèi)中肯定沒(méi)有提供對(duì)外可setters方法。接下來(lái)來(lái)具體看一下String類(lèi)的定義须床。

下面是String類(lèi)中主要屬性的定義(Java 1.7源碼):

public final class String implements java.io.Serializable, Comparable<String>, CharSequence{
    
    /** The value is used for character storage. */
    private final char value[];
    
    /** Cache the hash code for the string */
    private int hash; // Default to 0
    
}

與之前版本的Java String源碼相比铐料,String類(lèi)減少了int offset 和 int count的定義。這樣變化的結(jié)果主要體現(xiàn)在:

  • 避免之前版本的String對(duì)象subString時(shí)可能引起的內(nèi)存泄露問(wèn)題豺旬;
  • 新版本的subString時(shí)間復(fù)雜度將有O(1)變?yōu)镺(n);

通過(guò)上面String類(lèi)的定義钠惩,類(lèi)名前面用了final class修飾,因此族阅,String類(lèi)不能被繼承篓跛。對(duì)于其屬性定義,可以看出坦刀,屬性value[]和hash都是被定義成private類(lèi)型愧沟,且由于沒(méi)有提供對(duì)外的public setters方法,String類(lèi)屬性不可被改變鲤遥。

其中沐寺,需要重點(diǎn)關(guān)注屬性value[],其被final char修飾盖奈,因此字符型數(shù)組value只會(huì)被賦值一次就不可修改混坞。其存儲(chǔ)內(nèi)容正好是String中的單個(gè)字符內(nèi)容。

2 常量池

JVM為了提高性能和減少內(nèi)存開(kāi)銷(xiāo)钢坦,內(nèi)部維護(hù)了一個(gè)字符串常量池究孕,每當(dāng)創(chuàng)建字符串常量時(shí),JVM首先檢查字符串常量池爹凹,如果常量池中已經(jīng)存在蚊俺,則返回池中的字符串對(duì)象引用,否則創(chuàng)建該字符串對(duì)象并放入池中逛万。

因此下述結(jié)果返回true。

String a = "abc";
String b = "abc";
System.out.print(a == b); //true

但與創(chuàng)建字符串常量方式不同的是,當(dāng)使用new String(String str)方式等創(chuàng)建字符串對(duì)象時(shí)宇植,不管字符串常量池中是否有與此相同內(nèi)容的字符串得封,都會(huì)在堆內(nèi)存中創(chuàng)建新的字符串對(duì)象。

因此指郁,下面代碼片段有如下結(jié)果忙上。

String a = "Hello";
String b = new String("Hello");
System.out.println(a == b);  //false
System.out.println(a.equals(b)); //true

即使字符串內(nèi)容相同,字符串常量池中的字符串與通過(guò)new String(..)等方式創(chuàng)建的字符串對(duì)象之間沒(méi)有直接的關(guān)系闲坎,但是疫粥,可以通過(guò)字符串的intern()方法找到此種關(guān)聯(lián)。

intern()方法返回字符串對(duì)象在字符串常量池中的對(duì)象引用腰懂,若字符串常量池中尚未有此字符串梗逮,則創(chuàng)建一新的字符串常量放置于池中。

于是绣溜,很據(jù)如上理解慷彤,很自然的,可以得到如下結(jié)果怖喻。

String a = "Hello";
System.out.println(a == a.intern()); //true

String b = new String("corn");
String c = b.intern();

System.out.println(b == c); //false

String d = "corn";

System.out.println(c == d); //true

3 String相關(guān) +

String中使用 + 字符串連接符進(jìn)行字符串連接時(shí)底哗,連接操作最開(kāi)始時(shí)如果都是字符串常量,編譯后將盡可能多的直接將字符串常量連接起來(lái)锚沸,形成新的字符串常量參與后續(xù)連接(通過(guò)反編譯工具jd-gui也可以方便的直接看出)跋选;

接下來(lái)的字符串連接是從左向右依次進(jìn)行,對(duì)于不同的字符串哗蜈,首先以最左邊的字符串為參數(shù)創(chuàng)建StringBuilder對(duì)象前标,然后依次對(duì)右邊進(jìn)行append操作,最后將StringBuilder對(duì)象通過(guò)toString()方法轉(zhuǎn)換成String對(duì)象(注意:中間的多個(gè)字符串常量不會(huì)自動(dòng)拼接)恬叹。

也就是說(shuō)

String c = "xx" + "yy " + a + "zz" + "mm" + b;

實(shí)質(zhì)上的實(shí)現(xiàn)過(guò)程是:

String c = new StringBuilder("xxyy").append(a).append("zz")
        .append("mm").append(b).toString();

由于得出結(jié)論:當(dāng)使用+進(jìn)行多個(gè)字符串連接時(shí)候生,實(shí)際上是產(chǎn)生了一個(gè)StringBuilder對(duì)象和一個(gè)String對(duì)象。

4 String/StringBuilder/StringBuffer區(qū)別

String是不可變字符串對(duì)象绽昼,StringBuilder和StringBuffer是可變字符串對(duì)象(其內(nèi)部的字符數(shù)組長(zhǎng)度可變)唯鸭,StringBuffer線程安全,StringBuilder非線程安全硅确。

5 既然String是不可變字符串對(duì)象目溉,如何才能改變讓其可變?

既然String對(duì)象中沒(méi)有對(duì)外提供可用的public setters等方法,因此只能通過(guò)Java中的反射機(jī)制實(shí)現(xiàn)菱农。因此缭付,前文中說(shuō)到的String是不可變字符串對(duì)象只是針對(duì)"正常情況下"。而非必然循未。

public static void stringReflection() throws Exception {

    String s = "Hello World";

    System.out.println("s = " + s); //Hello World

    //獲取String類(lèi)中的value字段
    Field valueField = String.class.getDeclaredField("value");

    //改變value屬性的訪問(wèn)權(quán)限
    valueField.setAccessible(true);

    char[] value = (char[]) valueField.get(s);

    //改變value所引用的數(shù)組中的第5個(gè)字符
    value[5] = '_';

    System.out.println("s = " + s); //Hello_World
}

6 String類(lèi)常用方法

  1. 求字符串長(zhǎng)度
    public int length()//返回該字符串的長(zhǎng)度
String str = new String("asdfzxc");
int strlength = str.length();//strlength = 7
  1. 求字符串某一位置字符
    public char charAt(int index)//返回字符串中指定位置的字符陷猫;注意字符串中第一個(gè)字符索引是0,最后一個(gè)是length()-1。
String str = new String("asdfzxc");
char ch = str.charAt(4);//ch = z
  1. 提取子串
    用String類(lèi)的substring方法可以提取字符串中的子串绣檬,該方法有兩種常用參數(shù):
    1)public String substring(int beginIndex)//該方法從beginIndex位置起足陨,從當(dāng)前字符串中取出剩余的字符作為一個(gè)新的字符串返回。
    2)public String substring(int beginIndex, int endIndex)//該方法從beginIndex位置起娇未,從當(dāng)前字符串中取出到endIndex-1位置的字符作為一個(gè)新的字符串返回墨缘。
String str1 = new String("asdfzxc");
String str2 = str1.substring(2);//str2 = "dfzxc"
String str3 = str1.substring(2,5);//str3 = "dfz"
  1. 字符串比較
    1)public int compareTo(String anotherString)//該方法是對(duì)字符串內(nèi)容按字典順序進(jìn)行大小比較,通過(guò)返回的整數(shù)值指明當(dāng)前字符串與參數(shù)字符串的大小關(guān)系零抬。若當(dāng)前對(duì)象比參數(shù)大則返回正整數(shù)镊讼,反之返回負(fù)整數(shù),相等返回0平夜。
    2)public int compareToIgnore(String anotherString)//與compareTo方法相似蝶棋,但忽略大小寫(xiě)。
    3)public boolean equals(Object anotherObject)//比較當(dāng)前字符串和參數(shù)字符串褥芒,在兩個(gè)字符串相等的時(shí)候返回true嚼松,否則返回false。
    4)public boolean equalsIgnoreCase(String anotherString)//與equals方法相似锰扶,但忽略大小寫(xiě)献酗。
String str1 = new String("abc");
String str2 = new String("ABC");
int a = str1.compareTo(str2);//a>0
int b = str1.compareTo(str2);//b=0
boolean c = str1.equals(str2);//c=false
boolean d = str1.equalsIgnoreCase(str2);//d=true
  1. 字符串連接
    public String concat(String str)//將參數(shù)中的字符串str連接到當(dāng)前字符串的后面,效果等價(jià)于"+"坷牛。
String str = "aa".concat("bb").concat("cc");
相當(dāng)于String str = "aa"+"bb"+"cc";
  1. 字符串中單個(gè)字符查找
    1)public int indexOf(int ch/String str)//用于查找當(dāng)前字符串中字符或子串罕偎,返回字符或子串在當(dāng)前字符串中從左邊起首次出現(xiàn)的位置,若沒(méi)有出現(xiàn)則返回-1京闰。
    2)public int indexOf(int ch/String str, int fromIndex)//改方法與第一種類(lèi)似颜及,區(qū)別在于該方法從fromIndex位置向后查找。
    3)public int lastIndexOf(int ch/String str)//該方法與第一種類(lèi)似蹂楣,區(qū)別在于該方法從字符串的末尾位置向前查找俏站。
    4)public int lastIndexOf(int ch/String str, int fromIndex)//該方法與第二種方法類(lèi)似,區(qū)別于該方法從fromIndex位置向前查找痊土。
String str = "I am a good student";
int a = str.indexOf('a');//a = 2
int b = str.indexOf("good");//b = 7
int c = str.indexOf("w",2);//c = -1
int d = str.lastIndexOf("a");//d = 5
int e = str.lastIndexOf("a",3);//e = 2
  1. 字符串中字符的大小寫(xiě)轉(zhuǎn)換
    1)public String toLowerCase()//返回將當(dāng)前字符串中所有字符轉(zhuǎn)換成小寫(xiě)后的新串
    2)public String toUpperCase()//返回將當(dāng)前字符串中所有字符轉(zhuǎn)換成大寫(xiě)后的新串
String str = new String("asDF");
String str1 = str.toLowerCase();//str1 = "asdf"
String str2 = str.toUpperCase();//str2 = "ASDF"
  1. 字符串中字符的替換
    1)public String replace(char oldChar, char newChar)//用字符newChar替換當(dāng)前字符串中所有的oldChar字符肄扎,并返回一個(gè)新的字符串。
    2)public String replaceFirst(String regex, String replacement)//該方法用字符replacement的內(nèi)容替換當(dāng)前字符串中遇到的第一個(gè)和字符串regex相匹配的子串赁酝,應(yīng)將新的字符串返回犯祠。
    3)public String replaceAll(String regex, String replacement)//該方法用字符replacement的內(nèi)容替換當(dāng)前字符串中遇到的所有和字符串regex相匹配的子串,應(yīng)將新的字符串返回酌呆。
String str = "asdzxcasd";
String str1 = str.replace('a','g');//str1 = "gsdzxcgsd"
String str2 = str.replace("asd","fgh");//str2 = "fghzxcfgh"
String str3 = str.replaceFirst("asd","fgh");//str3 = "fghzxcasd"
String str4 = str.replaceAll("asd","fgh");//str4 = "fghzxcfgh"

9衡载、其他類(lèi)方法
1)String trim()//截去字符串兩端的空格,但對(duì)于中間的空格不處理。

String str = " a sd ";
String str1 = str.trim();
int a = str.length();//a = 6
int b = str1.length();//b = 4

2)boolean statWith(String prefix)boolean endWith(String suffix)//用來(lái)比較當(dāng)前字符串的起始字符或子字符串prefix和終止字符或子字符串suffix是否和當(dāng)前字符串相同,重載方法中同時(shí)還可以指定比較的開(kāi)始位置offset。

String str = "asdfgh";
boolean a = str.statWith("as");//a = true
boolean b = str.endWith("gh");//b = true

3)regionMatches(boolean ignoreCase, int toffset, String other, int ooffset, int len)

  • ignoreCase -- 如果為 true简珠,則比較字符時(shí)忽略大小寫(xiě)猜揪。
  • toffset -- 此字符串中子區(qū)域的起始偏移量惭墓。
  • other -- 字符串參數(shù)。
  • ooffset -- 字符串參數(shù)中子區(qū)域的起始偏移量而姐。
  • len -- 要比較的字符數(shù)。
String Str1 = new String("www.runoob.com");
String Str2 = new String("runoob");
String Str3 = new String("RUNOOB");
String Str4 = new String("runaab");
String Str5 = new String("aaaoob");

System.out.println(Str1.regionMatches(4, Str2, 0, 5)); //true

System.out.println(Str1.regionMatches(4, Str3, 0, 5)); //false

System.out.println(Str1.regionMatches(true, 4, Str3, 0, 5)); //true

System.out.println(Str1.regionMatches(4, Str4, 0, 4)); //false

System.out.println(Str1.regionMatches(4, Str5, 2, 4)); //false

4)contains(String str)//判斷參數(shù)s是否被包含在字符串中划咐,并返回一個(gè)布爾類(lèi)型的值拴念。

String str = "student";
str.contains("stu");//true
str.contains("ok");//false

5)String[] split(String str)//將str作為分隔符進(jìn)行字符串分解,分解后的字字符串在字符串?dāng)?shù)組中返回褐缠。

String str = "asd!qwe|zxc#";
String[] str1 = str.split("!|#");
//str1[0] = "asd";str1[1] = "qwe";str1[2] = "zxc";

7 字符串與基本類(lèi)型的轉(zhuǎn)換

  1. 字符串轉(zhuǎn)換為基本類(lèi)型
    java.lang包中有Byte政鼠、Short、Integer队魏、Float公般、Double類(lèi)的調(diào)用方法:
    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)
    例如:
int n = Integer.parseInt("12");
float f = Float.parseFloat("12.34");
double d = Double.parseDouble("1.124");
  1. 基本類(lèi)型轉(zhuǎn)換為字符串類(lèi)型
    String類(lèi)中提供了String valueOf()放法,用作基本類(lèi)型轉(zhuǎn)換為字符串類(lèi)型胡桨。
    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)
    例如:
String s1 = String.valueOf(12);
String s1 = String.valueOf(12.34);
  1. 進(jìn)制轉(zhuǎn)換
    使用Long類(lèi)中的方法得到整數(shù)之間的各種進(jìn)制轉(zhuǎn)換的方法:
    Long.toBinaryString(long l)
    Long.toOctalString(long l)
    Long.toHexString(long l)
    Long.toString(long l, int p)//p作為任意進(jìn)制
?著作權(quán)歸作者所有,轉(zhuǎn)載或內(nèi)容合作請(qǐng)聯(lián)系作者
  • 序言:七十年代末官帘,一起剝皮案震驚了整個(gè)濱河市,隨后出現(xiàn)的幾起案子昧谊,更是在濱河造成了極大的恐慌刽虹,老刑警劉巖,帶你破解...
    沈念sama閱讀 218,036評(píng)論 6 506
  • 序言:濱河連續(xù)發(fā)生了三起死亡事件呢诬,死亡現(xiàn)場(chǎng)離奇詭異涌哲,居然都是意外死亡,警方通過(guò)查閱死者的電腦和手機(jī)尚镰,發(fā)現(xiàn)死者居然都...
    沈念sama閱讀 93,046評(píng)論 3 395
  • 文/潘曉璐 我一進(jìn)店門(mén)阀圾,熙熙樓的掌柜王于貴愁眉苦臉地迎上來(lái),“玉大人狗唉,你說(shuō)我怎么就攤上這事初烘。” “怎么了敞曹?”我有些...
    開(kāi)封第一講書(shū)人閱讀 164,411評(píng)論 0 354
  • 文/不壞的土叔 我叫張陵账月,是天一觀的道長(zhǎng)。 經(jīng)常有香客問(wèn)我澳迫,道長(zhǎng)局齿,這世上最難降的妖魔是什么? 我笑而不...
    開(kāi)封第一講書(shū)人閱讀 58,622評(píng)論 1 293
  • 正文 為了忘掉前任橄登,我火速辦了婚禮抓歼,結(jié)果婚禮上讥此,老公的妹妹穿的比我還像新娘。我一直安慰自己谣妻,他們只是感情好萄喳,可當(dāng)我...
    茶點(diǎn)故事閱讀 67,661評(píng)論 6 392
  • 文/花漫 我一把揭開(kāi)白布。 她就那樣靜靜地躺著蹋半,像睡著了一般他巨。 火紅的嫁衣襯著肌膚如雪。 梳的紋絲不亂的頭發(fā)上减江,一...
    開(kāi)封第一講書(shū)人閱讀 51,521評(píng)論 1 304
  • 那天染突,我揣著相機(jī)與錄音,去河邊找鬼辈灼。 笑死份企,一個(gè)胖子當(dāng)著我的面吹牛,可吹牛的內(nèi)容都是我干的巡莹。 我是一名探鬼主播司志,決...
    沈念sama閱讀 40,288評(píng)論 3 418
  • 文/蒼蘭香墨 我猛地睜開(kāi)眼,長(zhǎng)吁一口氣:“原來(lái)是場(chǎng)噩夢(mèng)啊……” “哼降宅!你這毒婦竟也來(lái)了骂远?” 一聲冷哼從身側(cè)響起,我...
    開(kāi)封第一講書(shū)人閱讀 39,200評(píng)論 0 276
  • 序言:老撾萬(wàn)榮一對(duì)情侶失蹤钉鸯,失蹤者是張志新(化名)和其女友劉穎吧史,沒(méi)想到半個(gè)月后,有當(dāng)?shù)厝嗽跇?shù)林里發(fā)現(xiàn)了一具尸體唠雕,經(jīng)...
    沈念sama閱讀 45,644評(píng)論 1 314
  • 正文 獨(dú)居荒郊野嶺守林人離奇死亡贸营,尸身上長(zhǎng)有42處帶血的膿包…… 初始之章·張勛 以下內(nèi)容為張勛視角 年9月15日...
    茶點(diǎn)故事閱讀 37,837評(píng)論 3 336
  • 正文 我和宋清朗相戀三年,在試婚紗的時(shí)候發(fā)現(xiàn)自己被綠了岩睁。 大學(xué)時(shí)的朋友給我發(fā)了我未婚夫和他白月光在一起吃飯的照片钞脂。...
    茶點(diǎn)故事閱讀 39,953評(píng)論 1 348
  • 序言:一個(gè)原本活蹦亂跳的男人離奇死亡,死狀恐怖捕儒,靈堂內(nèi)的尸體忽然破棺而出冰啃,到底是詐尸還是另有隱情,我是刑警寧澤刘莹,帶...
    沈念sama閱讀 35,673評(píng)論 5 346
  • 正文 年R本政府宣布阎毅,位于F島的核電站,受9級(jí)特大地震影響点弯,放射性物質(zhì)發(fā)生泄漏扇调。R本人自食惡果不足惜,卻給世界環(huán)境...
    茶點(diǎn)故事閱讀 41,281評(píng)論 3 329
  • 文/蒙蒙 一抢肛、第九天 我趴在偏房一處隱蔽的房頂上張望狼钮。 院中可真熱鬧碳柱,春花似錦、人聲如沸熬芜。這莊子的主人今日做“春日...
    開(kāi)封第一講書(shū)人閱讀 31,889評(píng)論 0 22
  • 文/蒼蘭香墨 我抬頭看了看天上的太陽(yáng)涎拉。三九已至瑞侮,卻和暖如春,著一層夾襖步出監(jiān)牢的瞬間曼库,已是汗流浹背区岗。 一陣腳步聲響...
    開(kāi)封第一講書(shū)人閱讀 33,011評(píng)論 1 269
  • 我被黑心中介騙來(lái)泰國(guó)打工, 沒(méi)想到剛下飛機(jī)就差點(diǎn)兒被人妖公主榨干…… 1. 我叫王不留毁枯,地道東北人。 一個(gè)月前我還...
    沈念sama閱讀 48,119評(píng)論 3 370
  • 正文 我出身青樓叮称,卻偏偏與公主長(zhǎng)得像种玛,于是被迫代替她去往敵國(guó)和親。 傳聞我的和親對(duì)象是個(gè)殘疾皇子瓤檐,可洞房花燭夜當(dāng)晚...
    茶點(diǎn)故事閱讀 44,901評(píng)論 2 355

推薦閱讀更多精彩內(nèi)容

  • 1. Java基礎(chǔ)部分 基礎(chǔ)部分的順序:基本語(yǔ)法赂韵,類(lèi)相關(guān)的語(yǔ)法,內(nèi)部類(lèi)的語(yǔ)法挠蛉,繼承相關(guān)的語(yǔ)法祭示,異常的語(yǔ)法,線程的語(yǔ)...
    子非魚(yú)_t_閱讀 31,631評(píng)論 18 399
  • 前面我們總結(jié)了數(shù)組操作谴古,這里我們將總結(jié)字符串相關(guān)的知識(shí)质涛,除了總結(jié)String的API用法,同時(shí)我們還會(huì)總結(jié)一些相關(guān)...
    HCherisher閱讀 3,621評(píng)論 2 6
  • 一掰担、String類(lèi) String類(lèi)在java.lang包中汇陆,java使用String類(lèi)創(chuàng)建一個(gè)字符串變量,字符串變...
    wlw_花田半畝閱讀 434評(píng)論 0 1
  • Spring Cloud為開(kāi)發(fā)人員提供了快速構(gòu)建分布式系統(tǒng)中一些常見(jiàn)模式的工具(例如配置管理带饱,服務(wù)發(fā)現(xiàn)毡代,斷路器,智...
    卡卡羅2017閱讀 134,656評(píng)論 18 139
  • 最近發(fā)生了太多太多的千奇百怪的事。好的壞的执庐。積極的酪耕,墮落的,傷心的耕肩,開(kāi)心的因妇。一點(diǎn)一滴我都不知道如何梳理眾多的情緒集...
    怡然自得的先知閱讀 554評(píng)論 3 4