String實(shí)例化及static final修飾符實(shí)現(xiàn)方法解析
String兩種實(shí)例化方式
一種是通過雙引號(hào)直接賦值的方式,另外一種是使用標(biāo)準(zhǔn)的new調(diào)用構(gòu)造方法完成實(shí)例化。如下:
String str = "abcd";
String str = new String("1234);
第一種方法:
使用直接賦值后,只要是以后聲明的字符串內(nèi)容相同,則不會(huì)再開辟新的內(nèi)存空間。對(duì)于String的以上操作,在java中稱為共享設(shè)計(jì)。這種設(shè)計(jì)思路是,在java中形成一個(gè)字符串對(duì)象池,在這個(gè)字符串對(duì)象中保存多個(gè)字符串對(duì)象,新實(shí)例化的對(duì)象如果已經(jīng)在池中定義了,則不再重新定義,而從池中直接取出繼續(xù)使用。String就是因?yàn)椴捎昧诉@樣的設(shè)計(jì),所以當(dāng)內(nèi)容重復(fù)時(shí),會(huì)將對(duì)象指向已存在的實(shí)例空間。
一個(gè)雙引號(hào)包含字符串就是一個(gè)String類的匿名對(duì)象,但是這種方式使用String不一定創(chuàng)建新對(duì)象。在執(zhí)行到這個(gè)字符串的語句時(shí),如String a = "123",JVM會(huì)先到常量池里查找,如果有的話返回常量池里的這個(gè)實(shí)例的引用,否則的話創(chuàng)建一個(gè)新實(shí)例并置入常量池里。
第二種方法:
使用new關(guān)鍵字,不管如何都會(huì)再開辟一個(gè)新的空間。
new創(chuàng)建字符串時(shí)首先查看池中是否有相同值的字符串,如果有,則拷貝一份到堆中,然后返回堆中的地址;如果池中沒有,則在堆中創(chuàng)建一份,然后返回堆中的地址(注意,此時(shí)不需要從堆中復(fù)制到池中,否則,將使得堆中的字符串永遠(yuǎn)是池中的子集,導(dǎo)致浪費(fèi)池的空間)!
String實(shí)例化的時(shí)機(jī)
(1)單獨(dú)使用""引號(hào)創(chuàng)建的字符串都是常量,編譯期就已經(jīng)確定存儲(chǔ)到String Pool中;
(2)使用new String("")創(chuàng)建的對(duì)象會(huì)存儲(chǔ)到堆區(qū)(heap)中,是運(yùn)行期新創(chuàng)建的;
(3)使用只包含常量的字符串連接符如"aa" + "aa"創(chuàng)建的也是常量,編譯期就能確定,已經(jīng)確定存儲(chǔ)到String Pool中;
(4)使用包含變量的字符串連接符如"aa" + s1創(chuàng)建的對(duì)象是運(yùn)行期才創(chuàng)建的,存儲(chǔ)在堆區(qū)(heap)中;
注意:上面第(3)句話,編譯后合并的字符串會(huì)保存在JVM的字符串池中,而不是再生成的class文件中把字符串合并。
String s = "a" + "b" + "c"; 創(chuàng)建的是一個(gè)對(duì)象,而不是是四個(gè)對(duì)象,在字符串常量池中只生成一個(gè)字符串對(duì)象
字符串池的優(yōu)缺點(diǎn)
字符串池的優(yōu)點(diǎn)就是避免了相同內(nèi)容的字符串的創(chuàng)建,節(jié)省了內(nèi)存,省去了創(chuàng)建相同字符串的時(shí)間,同時(shí)提升了性能;另一方面,字符串池的缺點(diǎn)就是犧牲了JVM在常量池中遍歷對(duì)象所需要的時(shí)間,不過其時(shí)間成本相比而言比較低。
static final修飾的字符串好嗎?
工作后發(fā)現(xiàn),大型的項(xiàng)目里,常常會(huì)見到定義字符串使用 private static final String = "abc" 的方式。這種方式有好處嗎?
首先使用直接賦值的字串的方式,字符串會(huì)在編譯期生成在字符串池中。
然后final標(biāo)記的變量(成員變量或局部變量)即成為常量,只能賦值一次。它應(yīng)該不影響內(nèi)存的分配。(查看資料多了,說法不一,在下對(duì)此也有點(diǎn)懷疑了,如果final影響內(nèi)存分配,煩請(qǐng)各位大俠告知)
最后看static修飾符:
static修飾符能夠與屬性、方法和內(nèi)部類一起使用,表示靜態(tài)的。類中的靜態(tài)變量和靜態(tài)方法能夠與類名一起使用,不需要?jiǎng)?chuàng)建一個(gè)類的對(duì)象來訪問該類的靜態(tài)成員,所以,static修飾的變量又稱作“類變量”。
“類變量”屬于類的成員,類的成員是被儲(chǔ)存在堆內(nèi)存里面的。一個(gè)類中,一個(gè)static變量只會(huì)有一個(gè)內(nèi)存空間,即使有多個(gè)類實(shí)例,但這些類實(shí)例中的這個(gè)static變量會(huì)共享同一個(gè)內(nèi)存空間。
static修飾的String,會(huì)在堆內(nèi)存中復(fù)制一份常量池中的值。所以調(diào)用 static final String 變量,實(shí)際上是直接調(diào)用堆內(nèi)存的地址,不會(huì)遍歷字符串池中的對(duì)象,節(jié)省了遍歷時(shí)間。
所以使用static final修飾的字符串還是有好處的。
代碼測(cè)試
public class Test { public static final String A="ab"; public static final String B="cd"; public static final String C; public static final String D; static{ C = "ab"; D = "cd"; } public static void main(String[] args) { String t = "abcd";//指向池 String s1 = "ab";//指向池 String s2 = "cd";//指向池 String s = s1+s2;//指向堆 System.out.println(s==t);//false String ss = "ab"+s2;//指向堆 System.out.println(ss==t);//false String sss = "ab"+"cd";//指向池 System.out.println(sss==t);//true String ssss = A+B;//指向池 System.out.println(ssss==t);//true System.out.println((C+D)==t);//false } }
字符串對(duì)象可以存放在兩個(gè)地方,字符串池(pool)和堆,編譯期確定如何給一個(gè)引用變量賦值
- String s="abc";這種形式?jīng)Q定將從pool中尋找內(nèi)容相同的字符串并返回地址給s,pool中沒有就會(huì)在pool中新建并返回地址給s
- String s = new String("abc");這種形式?jīng)Q定運(yùn)行期將在堆上新建字符串對(duì)象并返回給s,但這個(gè)對(duì)象不會(huì)加入到pool中
- String s=s1+s2;s1和s2都是變量,這種形式?jīng)Q定將在堆上創(chuàng)建s1和s2(即便s1和s2指向的對(duì)象在池中已經(jīng)存在,也會(huì)將值拷貝到對(duì)象創(chuàng)建新對(duì)象),然后創(chuàng)建s1+s2并賦給s
- String s = "ab"+"cd";同1),都是來自于池
- String s = "ab"+s1;類似3)
- String s = S1+S2;S1和S2是常量,常量只能賦值一次,S1,S2如果在聲明的地方就賦值,那么這個(gè)值在編譯期就是確定的,后面無法更改,S1+S2在執(zhí)行前可確定S1/S2已經(jīng)在池中存在,當(dāng)然在池中進(jìn)行,所以s指向pool;但是若S1,S2如果是實(shí)例常量在構(gòu)造器中賦值,或是類常量在靜態(tài)塊中賦值,S1+S2無法確定二者皆來自于池,于是在堆上進(jìn)行
以上就是本文的全部內(nèi)容,希望對(duì)大家的學(xué)習(xí)有所幫助,也希望大家多多支持腳本之家。
- Node.js API詳解之 querystring用法實(shí)例分析
- Node.js API詳解之 string_decoder用法實(shí)例分析
- Java string類型轉(zhuǎn)換成map代碼實(shí)例
- java將String字符串轉(zhuǎn)換為List<Long>類型實(shí)例方法
- Python將string轉(zhuǎn)換到float的實(shí)例方法
- php類中的$this,static,final,const,self這幾個(gè)關(guān)鍵字使用方法
- Java修飾符 abstract,static,final 的區(qū)別詳解
- php面向?qū)ο笕ヂ?(十) final static const關(guān)鍵字的使用
相關(guān)文章
Java調(diào)用IK分詞器進(jìn)行分詞方式,封裝工具類
這篇文章主要介紹了Java調(diào)用IK分詞器進(jìn)行分詞方式,封裝工具類,具有很好的參考價(jià)值,希望對(duì)大家有所幫助,如有錯(cuò)誤或未考慮完全的地方,望不吝賜教2024-08-08SpringBoot使用spring.config.import多種方式導(dǎo)入配置文件
本文主要介紹了SpringBoot使用spring.config.import多種方式導(dǎo)入配置文件,文中通過示例代碼介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友們下面隨著小編來一起學(xué)習(xí)學(xué)習(xí)吧2022-05-05Java HashSet(散列集),HashMap(散列映射)的簡單介紹
這篇文章主要介紹了Java HashSet(散列集),HashMap(散列映射)的簡單介紹,幫助大家更好的理解和學(xué)習(xí)Java集合框架的相關(guān)知識(shí),感興趣的朋友可以了解下2021-01-01Java Ehcache緩存框架入門級(jí)使用實(shí)例
這篇文章主要介紹了Java Ehcache緩存框架入門級(jí)使用實(shí)例,具有一定的參考價(jià)值,感興趣的小伙伴們可以參考一下。2017-08-08springboot接收http請(qǐng)求,解決參數(shù)中+號(hào)變成空格的問題
這篇文章主要介紹了springboot接收http請(qǐng)求,解決參數(shù)中+號(hào)變成空格的問題,具有很好的參考價(jià)值,希望對(duì)大家有所幫助。如有錯(cuò)誤或未考慮完全的地方,望不吝賜教2021-08-08詳解在Java程序中運(yùn)用Redis緩存對(duì)象的方法
這篇文章主要介紹了在Java程序中運(yùn)用Redis緩存對(duì)象的方法,文中通過示例代碼介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友們下面隨著小編來一起學(xué)習(xí)學(xué)習(xí)吧2019-03-03