code iban 是有什么组成_「面试」new String(abc)和abc有什么区别?反编译看看原理吧...
以下內(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"
代碼中"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
這里的樣例和上一個樣例存在一些差別,這里的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
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 == s2和String 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)容,希望文章能夠幫你解決所遇到的問題。
- 上一篇: android信鸽推送demo_【厚积薄
- 下一篇: 444 nginx_nginx 安全问题