日韩性视频-久久久蜜桃-www中文字幕-在线中文字幕av-亚洲欧美一区二区三区四区-撸久久-香蕉视频一区-久久无码精品丰满人妻-国产高潮av-激情福利社-日韩av网址大全-国产精品久久999-日本五十路在线-性欧美在线-久久99精品波多结衣一区-男女午夜免费视频-黑人极品ⅴideos精品欧美棵-人人妻人人澡人人爽精品欧美一区-日韩一区在线看-欧美a级在线免费观看

歡迎訪問 生活随笔!

生活随笔

當(dāng)前位置: 首頁 > 编程资源 > 编程问答 >内容正文

编程问答

code iban 是有什么组成_「面试」new String(abc)和abc有什么区别?反编译看看原理吧...

發(fā)布時間:2025/3/20 编程问答 23 豆豆
生活随笔 收集整理的這篇文章主要介紹了 code iban 是有什么组成_「面试」new String(abc)和abc有什么区别?反编译看看原理吧... 小編覺得挺不錯的,現(xiàn)在分享給大家,幫大家做個參考.

以下內(nèi)容均由本人獨立完成,希望你看完之后能有更多更深入的了解,歡迎關(guān)注?~

在學(xué)習(xí)String類之前先看看如下的代碼塊

public String gets1() { return "" + System.currentTimeMillis();}public String gets2() { return String.valueOf(System.currentTimeMillis());}public String gets3() { return new String("") + System.currentTimeMillis();}

很顯然三個方法都是輸出當(dāng)前時間戳string類型的數(shù)據(jù),那么他們有什么不同呢?反編譯得出三個方法的操作細(xì)節(jié)

public java.lang.String gets1();Code: 0: new #2 // class java/lang/StringBuilder 3: dup 4: invokespecial #3 // Method java/lang/StringBuilder."":()V 7: ldc #4 // String 9: invokevirtual #5 // Method java/lang/StringBuilder.append:(Ljava/lang/String;)Ljava/lang/StringBuilder; 12: invokestatic #6 // Method java/lang/System.currentTimeMillis:()J 15: invokevirtual #7 // Method java/lang/StringBuilder.append:(J)Ljava/lang/StringBuilder; 18: invokevirtual #8 // Method java/lang/StringBuilder.toString:()Ljava/lang/String; 21: areturn' public java.lang.String gets2();Code: 0: invokestatic #6 // Method java/lang/System.currentTimeMillis:()J 3: invokestatic #9 // Method java/lang/String.valueOf:(J)Ljava/lang/String; 6: areturnpublic java.lang.String gets3();Code: 0: new #2 // class java/lang/StringBuilder 3: dup 4: invokespecial #3 // Method java/lang/StringBuilder."":()V 7: new #10 // class java/lang/String 10: dup 11: ldc #4 // String 13: invokespecial #11 // Method java/lang/String."":(Ljava/lang/String;)V 16: invokevirtual #5 // Method java/lang/StringBuilder.append:(Ljava/lang/String;)Ljava/lang/StringBuilder; 19: invokestatic #6 // Method java/lang/System.currentTimeMillis:()J 22: invokevirtual #7 // Method java/lang/StringBuilder.append:(J)Ljava/lang/StringBuilder; 25: invokevirtual #8 // Method java/lang/StringBuilder.toString:()Ljava/lang/String; 28: areturn
  • 方法1:"" + System.currentTimeMillis() 是通過StringBuilder的append方法添加了一個空字符串以及當(dāng)前時間戳,然后通過toString方法生成一個全新的String對象。使用了一個StringBuilder對象,一個String對象
  • 方法2:String.valueOf(System.currentTimeMillis()) 直接獲取到當(dāng)前時間戳,然后創(chuàng)建一個String對象。使用了一個String對象
  • 方法3:new String("") + System.currentTimeMillis() 則最麻煩,先創(chuàng)建了StringBuilder對象,然后新建一個為空字符串的string對象,通過append方法添加進(jìn)去,最后toString創(chuàng)建一個新的String對象。使用了一個StringBuilder對象,兩個String對象

很明顯上述三個方法中方法2是最優(yōu)的,那么這是為什么呢?接下來就深入學(xué)習(xí)一下String 類。在學(xué)習(xí)之前先了解下JVM的組成結(jié)構(gòu)

  • 堆:jvm運行中申請的對象存放的位置。也就是所說的新生代+老年代,YGC就發(fā)生在這里
  • 虛擬機(jī)棧:每個方法在被調(diào)用執(zhí)行時都會創(chuàng)建一個虛擬機(jī)棧,用于存儲臨時的遍歷、方法等信息。調(diào)用相當(dāng)于進(jìn)棧,返回結(jié)果則相當(dāng)于出棧,異常輸出的棧信息就是從這里來的。
  • 本地方法棧:也是方法調(diào)用,只是調(diào)用的方法是本地native方法
  • 方法區(qū):存儲的類的結(jié)構(gòu)信息,靜態(tài)變量等信息。也就是永生代,發(fā)生FULL GC的地方(Java8中以元空間代替了永生代這個概念)
  • 常量池:方法區(qū)的一部分,用來存放各種生成的字面量,例如定義的一個String類型的數(shù)據(jù)(Java7中把常量池移到了堆中)

1、創(chuàng)建字符串

創(chuàng)建字符串一般就兩種方法

  • String s1="hello"
  • String s2=new String("hello")

第一種方法是在棧中創(chuàng)建一個String類型的引用s1,然后在常量池中尋找,如果常量池中存在hello的字符串?dāng)?shù)據(jù),則直接把s1指向常量池中hello的地址;否則會在常量池中創(chuàng)建hello這個字符串?dāng)?shù)據(jù),然后把s1指向新創(chuàng)建的hello地址。

第二種方法是在棧中創(chuàng)建一個String類型的引用s2,然后在常量池中尋找,如果常量池中存在該數(shù)據(jù),則在堆中復(fù)制拷貝該數(shù)據(jù),然后把s2指向堆中新建的地址;否則會創(chuàng)建一個字符串存放在常量池中,然后進(jìn)行拷貝一份到堆中,s2指向堆的地址。

通過new創(chuàng)建的string一定會在堆中創(chuàng)建一份數(shù)據(jù),同時常量池肯定有一個值的備份操作;而單獨的字符串則是直接指向常量池,所以一般還是使用字符串更好些,具體可看如下的圖解

創(chuàng)建字符串

2、字符串 + 操作

  • String s1="helloworld"
  • String s2="hello" + "world"
  • String s3=new String("hello") + "world"
System.out.println(s1 == s2); // trueSystem.out.println(s2 == s3); // falseSystem.out.println(s1 == s3); // false

代碼中"hello" + "world"在編譯期已經(jīng)知道了數(shù)據(jù)情況(使用javac編譯查看class文件會發(fā)現(xiàn)s1和s2是一致的),JVM會自動優(yōu)化使得s1和s2是一樣的,s3由于有new操作,所以需要StringBuilder的append完成,具體可看如下圖解。

字符串 + 操作

3、字符串變量 + 操作

  • String s1="hello"
  • String s2="world"
  • String s3="helloworld"
  • String s4=s1 + s2
System.out.println(s3 == s4); // false

這里的樣例和上一個樣例存在一些差別,這里的s4是由s1 + s2獲得的,在編譯期無法感知到其實際值,在運行期時會利用StringBuilder的append剩下一個新的String對象,所以s3指向的是常量池,而s4指向的堆,兩者自然是不一樣的。

s4 = new StringBuilder()).append(s1).append(s2).toString()

4、帶final的字符串變量 + 操作

  • final String s1="hello"
  • String s2="world"
  • String s3="helloworld"
  • String s4=s1 + "world"
  • String s5=s1 + s2
System.out.println(s3 == s4); // trueSystem.out.println(s3 == s5); // false

javac編譯之后的class文件如下

String var1 = "world";String var2 = "helloworld";String var3 = "helloworld";String var4 = "hello" + var1;System.out.println(var2 == var3);System.out.println(var2 == var4);

添加了final關(guān)鍵字修飾的變量在編譯期會被對應(yīng)的字符串直接替換掉,相當(dāng)于字符串?dāng)?shù)據(jù),而包含了字符串變量的+操作則依舊是使用了StringBuilder

5、intern() 方法

String.intern方法是一個native方法,獲取的是當(dāng)前字符串在常量池的數(shù)據(jù),如果常量池存在該數(shù)據(jù)則直接返回,如果不存在則把該數(shù)據(jù)添加到常量池中后返回,所以有String s1 = "abc" 和 String s2 = s1.intern() 中的s1 == s2String s1="world" 和 String s2=new String("world") 中 s1.intern() == s2.intern()

總結(jié)

String 本身是final類型的類,在日常使用中需要頻繁的做字符串合并操作時,盡可能的使用StringBuilder(如需要考慮線程安全則使用StringBuffer),降低無謂的字符串創(chuàng)建操作,在保證安全的情況下,提高效率!

總結(jié)

以上是生活随笔為你收集整理的code iban 是有什么组成_「面试」new String(abc)和abc有什么区别?反编译看看原理吧...的全部內(nèi)容,希望文章能夠幫你解決所遇到的問題。

如果覺得生活随笔網(wǎng)站內(nèi)容還不錯,歡迎將生活随笔推薦給好友。