一、前言
字符串是多個(gè)字符連接起來(lái)組合成的字符序列。字符串分為可變的字符串和不可變的字符串兩種。
(1)不可變的字符串:當(dāng)字符串對(duì)象創(chuàng)建完畢之后,該對(duì)象的內(nèi)容(上述的字符序列)是不能改變的
,一旦內(nèi)容改變就會(huì)創(chuàng)建一個(gè)新的字符串對(duì)象
。Java中的String類的對(duì)象就是不可變的。
(2)可變的字符串:StringBuilder類和StringBuffer類的對(duì)象就是可變的;當(dāng)對(duì)象創(chuàng)建完畢之后,該對(duì)象的內(nèi)容發(fā)生改變時(shí)不會(huì)創(chuàng)建新的對(duì)象,也就是說對(duì)象的內(nèi)容可以發(fā)生改變,當(dāng)對(duì)象的內(nèi)容發(fā)生改變時(shí),對(duì)象保持不變,還是同一個(gè)。
String、StringBuffer、StringBuilder 都實(shí)現(xiàn)了 CharSequence 接口,字符串在底層其實(shí)就是char[]
,雖然它們都與字符串相關(guān),但是其處理機(jī)制不同。
二、String 類(字符串常量)
String類表示不可變的字符串,當(dāng)前String類對(duì)象創(chuàng)建完畢之后,該對(duì)象的內(nèi)容(字符序列)是不變的,因?yàn)閮?nèi)容一旦改變就會(huì)創(chuàng)建一個(gè)一個(gè)新的對(duì)象。
String 類是final類,不可以繼承。
對(duì)String類型最好的重用方式是組合而不是繼承。
2.1 String 類實(shí)例的創(chuàng)建
方式一:通過字面量賦值創(chuàng)建
,需要注意這里是雙引號(hào):"",區(qū)別與字符char類型的單引號(hào):'';
String s1 = "laofu";
方式二:通過構(gòu)造器創(chuàng)建
;
String s2 = new String(“l(fā)aofu”);
兩種方式的區(qū)別:
方式一:String s1 = “l(fā)aofu”;
有可能只創(chuàng)建一個(gè)String對(duì)象,也有可能創(chuàng)建不創(chuàng)建String對(duì)象
;如果在常量池中已經(jīng)存在”laofu”,那么對(duì)象s1會(huì)直接引用,不會(huì)創(chuàng)建新的String對(duì)象;否則,會(huì)先在常量池先創(chuàng)建常量”laofu”的內(nèi)存空間,然后再引用。
方式二:String s2 = new String(“l(fā)aofu”);
最多會(huì)創(chuàng)建兩個(gè)String對(duì)象,最少創(chuàng)建一個(gè)String對(duì)象
??墒褂胣ew關(guān)鍵字創(chuàng)建對(duì)象是會(huì)在堆空間創(chuàng)建內(nèi)存區(qū)域,這是第一個(gè)對(duì)象;然后對(duì)象中的字符串字面量可能會(huì)創(chuàng)建第二個(gè)對(duì)象,而第二個(gè)對(duì)象如方式一中所描述的那樣,是有可能會(huì)不被創(chuàng)建的,所以至少創(chuàng)建一個(gè)String個(gè)對(duì)象。
上圖中的常量池:用于存儲(chǔ)常量的地方內(nèi)存區(qū)域,位于方法區(qū)中。常量池又分為編譯常量池和運(yùn)行常量池兩種:
編譯常量池:當(dāng)把字節(jié)碼加載進(jìn)JVM的時(shí)候,其中存儲(chǔ)的是字節(jié)碼的相關(guān)信息(如:行號(hào)等)。
運(yùn)行常量池:其中存儲(chǔ)的是代碼中的常量數(shù)據(jù)。
① 使用字符串字面量創(chuàng)建的字符串,也就是單獨(dú)使用""引號(hào)創(chuàng)建的字符串都是直接量,在編譯期就會(huì)將其存儲(chǔ)到常量池中;
② 使用new String("")創(chuàng)建的對(duì)象會(huì)存儲(chǔ)到堆內(nèi)存中,在運(yùn)行期才創(chuàng)建;
③ 使用只包含直接量的字符串連接符如"aa" + "bb"創(chuàng)建的也是直接量,這樣的字符串在編譯期就能確定,所以也會(huì)存儲(chǔ)到常量池中;
④ 使用包含String直接量的字符串表達(dá)式(如"aa" + s1)創(chuàng)建的對(duì)象是運(yùn)行期才創(chuàng)建的,對(duì)象存儲(chǔ)在堆中,因?yàn)槠涞讓邮莿?chuàng)新了StringBuilder對(duì)象來(lái)實(shí)現(xiàn)拼接的;
2.2 String 對(duì)象的比較
① 使用”==”號(hào):用于比較對(duì)象引用的內(nèi)存地址是否相同
② 使用equals方法:在Object類中和”==”號(hào)相同,但在自定義類中,建議覆蓋equals方法去實(shí)現(xiàn)比較自己內(nèi)容的細(xì)節(jié);由于String類覆蓋已經(jīng)覆蓋了equals方法,所以其比較的是字符串內(nèi)容
。
2.3 String對(duì)象的空值
① 對(duì)象引用為空, 此時(shí)s1沒有初始化,也在JVM中沒有分配內(nèi)存空間。
String s1 = null;
② 對(duì)象內(nèi)容為空字符串, 比如: 此時(shí)對(duì)象s2已經(jīng)初始化,值為“”,JVM已經(jīng)為其分配內(nèi)存空間。
String s2 = "";
2.4 字符串拼接
Java中的字符串可以通過是“+”實(shí)現(xiàn)拼接,那么代碼中字符串拼接在JVM中又是如何處理的呢?我們通過一個(gè)例子說明:通過比較拼接字符串代碼編譯前后的代碼來(lái)查看JVM對(duì)字符串拼接的處理。
JVM會(huì)對(duì)字符串拼接做一些優(yōu)化操作。
① 如果字符串字面量之間的拼接(如"aa" + “bb”),創(chuàng)建的也是直接量,這種情況在編譯期就能確定,所以也會(huì)存儲(chǔ)到常量池中
;
② 如果是對(duì)象之間拼接,或者是對(duì)象和字面量之間的拼接,亦或是方法執(zhí)行結(jié)果參與拼接,String內(nèi)部會(huì)使用StringBuilder先來(lái)獲取對(duì)象的值,然后使用append方法來(lái)執(zhí)行拼接
。這種情況只能在運(yùn)行期才能確定
變量的值和方法的返回值。
三、StringBuilder 與 StringBuffer(字符串變量)
StringBuffer 和 StringBuilder都表示可變的字符串,兩種的功能方法都是相同的。但唯一的區(qū)別:
(1)StringBuffer:StringBuffer中的方法都使用了synchronized修飾符,表示同步操作,在多線程并發(fā)的時(shí)候可以保證線程安全
,但在保證線程安全的時(shí)候,對(duì)其性能有一定影響,會(huì)降低其性能
。
(2)StringBuilder:StringBuilder中的方法都沒有使用了synchronized修飾符,線程不安全
,正因?yàn)槿绱?,?code>性能較高。
對(duì)并發(fā)安全沒有很高要求的情況下,建議使用StringBuilder,因?yàn)槠湫阅芎芨摺?/p>
四、String、StringBuilder 與 StringBuffer
(1)由于 String 類的操作是產(chǎn)生新的 String 對(duì)象,而 StringBuilder 和 StringBuffer 只是一個(gè)字符數(shù)組的擴(kuò)容而已,所以 String 類的操作要遠(yuǎn)慢于 StringBuffer 和 StringBuilder。
大部分情況下:StringBuilder > StringBuffer > String
String 類型和 StringBuffer 類型的主要性能區(qū)別其實(shí)在于 String 是不可變的對(duì)象, 因此在每次對(duì) String類型進(jìn)行改變的時(shí)候其實(shí)都等同于生成了一個(gè)新的 String 對(duì)象,然后將指針指向新的 String 對(duì)象。每次生成對(duì)象都會(huì)對(duì)系統(tǒng)性能產(chǎn)生影響,特別當(dāng)內(nèi)存中無(wú)引用對(duì)象多了以后, JVM 的 GC 就會(huì)開始工作,那速度是一定會(huì)相當(dāng)慢的。
而如果是使用 StringBuffer 類則結(jié)果就不一樣了,每次結(jié)果都會(huì)對(duì) StringBuffer 對(duì)象本身進(jìn)行操作,而不是生成新的對(duì)象,再改變對(duì)象引用。
(2)使用選擇
使用 String 類的場(chǎng)景:在字符串不經(jīng)常變化的場(chǎng)景中可以使用 String 類
,例如常量的聲明、少量的變量運(yùn)算。
使用 StringBuffer 類的場(chǎng)景:在頻繁進(jìn)行字符串運(yùn)算(如拼接、替換、刪除等),并且運(yùn)行在多線程環(huán)境中,則可以考慮使用 StringBuffer
,例如 XML 解析、HTTP 參數(shù)解析和封裝。
使用 StringBuilder 類的場(chǎng)景:在頻繁進(jìn)行字符串運(yùn)算(如拼接、替換、和刪除等),并且運(yùn)行在單線程的環(huán)境中,則可以考慮使用 StringBuilder
,如 SQL 語(yǔ)句的拼裝、JSON 封裝等。
本篇關(guān)于 Java 基礎(chǔ)知識(shí)之字符串 String 的知識(shí)總結(jié)的全部?jī)?nèi)容到此就介紹完了,想要了解更多關(guān)于 Java 字符串的其他內(nèi)容,請(qǐng)搜索W3Cschool相關(guān)技術(shù)文章,也希望大家可以多多關(guān)注和支持我們!