bl双性强迫侵犯h_国产在线观看人成激情视频_蜜芽188_被诱拐的少孩全彩啪啪漫画

Java字符串常量池和字面量賦值的簡單介紹

這篇文章主要介紹“Java字符串常量池和字面量賦值的簡單介紹”,在日常操作中,相信很多人在Java字符串常量池和字面量賦值的簡單介紹問題上存在疑惑,小編查閱了各式資料,整理出簡單好用的操作方法,希望對大家解答”Java字符串常量池和字面量賦值的簡單介紹”的疑惑有所幫助!接下來,請跟著小編一起來學習吧!

創新互聯公司長期為1000多家客戶提供的網站建設服務,團隊從業經驗10年,關注不同地域、不同群體,并針對不同對象提供差異化的產品和服務;打造開放共贏平臺,與合作伙伴共同營造健康的互聯網生態環境。為丹鳳企業提供專業的成都網站設計、網站建設,丹鳳網站改版等技術服務。擁有十余年豐富建站經驗和眾多成功案例,為您定制開發。

字符串常量池

之所以存在字符串常量池,主要是考慮到String字符串類型比八大基本類型更加復雜,并且使用的頻率也更加頻繁,經常創建字符串對象會造成性能瓶頸,所以采用相似度機制,將字符串進行復用(享元模式)。

在JDK 1.7 之后(包括1.7),字符串常量池已經從方法區移到了堆中。

字面量賦值

String s1 = "古時的風箏";

上面是字符串變量的最常用的方式,這種方式叫做字面量聲明,也就用把字符串用雙引號引起來,然后賦值給一個變量。這種情況下會直接將字符串放到字符串常量池中,然后返回給變量。

Java字符串常量池和字面量賦值的簡單介紹

那這是我再聲明一個內容相同的字符串,會發現字符串常量池中已經存在了,那直接指向常量池中的地址即可。

Java字符串常量池和字面量賦值的簡單介紹

例如上圖所示,聲明了 s1 和 s2,到最后都是指向同一個常量池的地址,所以 s1== s2 的結果是 true。

new String()方式

用 new String() 的方式,但是基本上不建議這么用,除非有特殊的邏輯需要。

String a = "古時的";
String s2 = new String(a + "風箏");

使用這種方式聲明字符串變量的時候,會有兩種情況發生,

字符串常量池之前已經存在相同字符串。

比如在使用 new 之前,已經用字面量聲明的方式聲明了一個變量,此時字符串常量池中已經存在了相同內容的字符串常量。

  • 首先會在堆中創建一個 s2 變量的對象引用;

  • 然后將這個對象引用指向字符串常量池中的已經存在的常量

Java字符串常量池和字面量賦值的簡單介紹

字符串常量池中不存在相同內容的常量

之前沒有任何地方用到了這個字符串,第一次聲明這個字符串就用的是 new String() 的方式,這種情況下會直接在堆中創建一個字符串對象然后返回給變量。

Java字符串常量池和字面量賦值的簡單介紹

我看到好多地方說,如果字符串常量池中不存在的話,就先把字符串先放進去,然后再引用字符串常量池的這個常量對象,這種說法是有問題的,只是 new String() 的話,如果池中沒有也不會放一份進去

基于 new String() 的這種特性,我們可以得出一個結論:

String s1 = "古時的風箏";
String a = "古時的";
String s2 = new String(a + "風箏");
String s3 = new String(a + "風箏");
System.out.println(s1==s2); // false
System.out.println(s2==s3);  // false

以上代碼,肯定輸出的都是 false,因為 new String() 不管你常量池中有沒有,我都會在堆中新建一個對象,新建出來的對象,當然不會和其他對象相等。

intern() 池化

那什么時候會放到字符串常量池呢,就是在使用 intern() 方法之后。 intern() 的定義:

  • 如果當前字符串內容存在于字符串常量池,存在的條件是使用 equas() 方法為true,也就是內容是一樣的,那直接返回此字符串在常量池的引用;

  • 如果之前不在字符串常量池中,那么在常量池創建一個引用并且指向堆中已存在的字符串,然后返回常量池中的地址

第一種情況,準備池化的字符串與字符串常量池中的字符串有相同(equas()判斷)

String s1 = "古時的風箏";
String a = "古時的";
String s2 = new String(a + "風箏");
s2 = s2.intern();

這個字符串常量已經在常量池存在了,這時,再 new 了一個新的對象 s2,并在堆中創建了一個相同字符串內容的對象。

Java字符串常量池和字面量賦值的簡單介紹

這時,s1 == s2 會返回 fasle。然后我們調用 s2 = s2.intern(),將池化操作返回的結果賦值給 s2,就會發生如下的變化。

Java字符串常量池和字面量賦值的簡單介紹

此時,再次判斷 s1 == s2 ,就會返回 true,因為它們都指向了字符串常量池的同一個字符串。

第二種情況,字符串常量池中不存在相同內容的字符串

使用 new String() 在堆中創建了一個字符串對象

Java字符串常量池和字面量賦值的簡單介紹

使用了intern()之后發生了什么呢,在常量池新增了一個對象,但是并沒有將字符串復制一份到常量池,而是直接指向了之前已經存在于堆中的字符串對象。

因為在 JDK 1.7 之后,字符串常量池不一定就是存字符串對象的,還有可能存儲的是一個指向堆中地址的引用,現在說的就是這種情況,注意了,下圖是只調用了 s2.intern(),并沒有返回給一個變量。其中字符串常量池(0x88)指向堆中字符串對象(0x99)就是intern() 的過程。

Java字符串常量池和字面量賦值的簡單介紹

只有當我們把 s2.intern() 的結果返回給 s2 時,s2 才真正的指向字符串常量池。

   public static void main(String[] args) {
 		String s1 = "古時的風箏";
		String s2 = "古時的風箏";
		String a = "古時的";
		String s3 = new String(a + "風箏");
		String s4 = new String(a + "風箏");
	    System.out.println(s1 == s2); // 【1】 true
	    System.out.println(s2 == s3); // 【2】 false
	    System.out.println(s3 == s4); // 【3】 false
	    s3.intern();
	    System.out.println(s2 == s3); // 【4】 false
	    s3 = s3.intern();
	    System.out.println(s2 == s3); // 【5】 true
	    s4 = s4.intern();
	    System.out.println(s3 == s4); // 【6】 true
  }

  • ① 使用字符串直接量時會在常量池創建對象,當然必須是常量折疊之后的。

  • ② 使用new String()時,new產生的字符串對象是位于堆中,而不是常量池中。

  • ③ JDK7之后intern()發生過變化,現在如果常量池中不存在這個對像,不會復制到常量池中,而是簡單的使用堆中已有字符串對象。

  • ④ JDK7以前的intern()不是這樣子的,以前會在常量池中創建一個新的對象,你可以將你的代碼,在JDK6中測試一下,結果應該會不同。所以,你的問題不在new String()上,而是在intern()上,前者與常量池從來就沒有關系。

到此,關于“Java字符串常量池和字面量賦值的簡單介紹”的學習就結束了,希望能夠解決大家的疑惑。理論與實踐的搭配能更好的幫助大家學習,快去試試吧!若想繼續學習更多相關知識,請繼續關注創新互聯網站,小編會繼續努力為大家帶來更多實用的文章!

當前文章:Java字符串常量池和字面量賦值的簡單介紹
分享地址:http://vcdvsql.cn/article14/gjdpde.html

成都網站建設公司_創新互聯,為您提供App設計云服務器微信公眾號網站導航用戶體驗企業網站制作

廣告

聲明:本網站發布的內容(圖片、視頻和文字)以用戶投稿、用戶轉載內容為主,如果涉及侵權請盡快告知,我們將會在第一時間刪除。文章觀點不代表本網站立場,如需處理請聯系客服。電話:028-86922220;郵箱:631063699@qq.com。內容未經允許不得轉載,或轉載時需注明來源: 創新互聯

成都seo排名網站優化