- 1、Java中String,StringBuffer和StringBuilder的區別和堆棧內存分配
- 2、JAVA里String數組在內存分配中分配的空間每個占幾個字節?
- 3、Java中 String字符串內存分配問題
- 4、Java StringBuilder和String的內存分析的問題
1.三者在執行速度方面的比較:StringBuilder StringBuffer String
2.String (StringBuffer,StringBuilder)的原因
String:字符串常量
StringBuffer:字符創變量
StringBuilder:字符創變量
從上面的名字可以看到,String是“字符創常量”,也就是不可改變的對象。對於這句話的理解你可能會產生這樣一個疑問 ,比如這段代碼:
String s = “abcd”;
s = s+1;
System.out.print(s);// result : abcd1
我們明明就是改變了String型的變量s的,為什麼說是沒有改變呢? 其實這是一種欺騙,JVM是這樣解析這段代碼的:首先創建對象s,賦予一個abcd,然後再創建一個新的對象s用來執行第二行代碼,也就是說我們之前對象s並沒有變化,所以我們說String類型是不可改變的對象了,由於這種機制,每當用String操作字符串時,實際上是在不斷的創建新的對象,而原來的對象就會變為垃圾被GC回收掉,可想而知這樣執行效率會有多底。
而StringBuffer與StringBuilder就不一樣了,他們是字符串變量,是可改變的對象,每當我們用它們對字符串做操作時,實際上是在一個對象上操作的,這樣就不會像String一樣創建一些而外的對象進行操作了,當然速度就快了。
3.一個特殊的例子:
String str = “This is only a” + “ simple” + “ test”;
StringBuffer builder = new StringBuilder(“This is only a”).append(“ simple”).append(“ test”);
你會很驚訝的發現,生成str對象的速度簡直太快了,而這個時候StringBuffer居然速度上根本一點都不佔優勢。其實這是JVM的一個把戲,實際上:
String str = “This is only a” + “ simple” + “test”;
其實就是:
String str = “This is only a simple test”;
所以不需要太多的時間了。但大家這裡要注意的是,如果你的字符串是來自另外的String對象的話,速度就沒那麼快了,譬如:
String str2 = “This is only a”;
String str3 = “ simple”;
String str4 = “ test”;
String str1 = str2 +str3 + str4;
這時候JVM會規規矩矩的按照原來的方式去做。
4.StringBuilder與 StringBuffer
StringBuilder:線程非安全的
StringBuffer:線程安全的
當我們在字符串緩衝去被多個線程使用是,JVM不能保證StringBuilder的操作是安全的,雖然他的速度最快,但是可以保證StringBuffer是可以正確操作的。當然大多數情況下就是我們是在單線程下進行的操作,所以大多數情況下是建議用StringBuilder而不用StringBuffer的,就是速度的原因。
對於三者使用的總結: 1.如果要操作少量的數據用 = String
2.單線程操作字符串緩衝區 下操作大量數據 = StringBuilder
3.多線程操作字符串緩衝區 下操作大量數據 = StringBuffer
java中一個int佔4個字節,由於java是unicode編碼,一個char可以佔兩個字節,可以存儲一個漢字。一個String,要看有幾個字母或者幾個漢字,一個英文字母就是一個字節,一個漢字就是兩個字節,以此類推計算。
jvm中有個String pool,一般在池中有的對象就不會再去生成一個新的。
String s1 = “abc”;
生成了一個字符串對象”abc”並放入pool中,定義了一個String變量s1並指向”abc”。
String s = “abc” + s1 + “def”;
等式右邊有三個String 對象 “abc” 、s1 、”def”
這裡(“abc”==s1)是true,就是說是引用的同一個內存地址。
“abc”已經在池中了,直接從池中取出來,這句又創建了一個”def”的String對象並放入池中。
當前二個String對象做連接的時候,又會創建一個新的String對象”abcabc”,並放入池中;然後再做字符串相加”abcabc”+def又生成一個字串對象,最後變量s指向這個新的String對象。
所以用+來連接字串會產生很多臨時的字串對象,效率低。一般使用StringBuffer或StringBuilder來做字串連接。
可以使用new String()來生成一個新的String對象並單獨分配內存空間。
——————————————————
查了下資料,已經改了回答內容。
String s = “abc”;
這句是定義一個String變量s,jvm會先去字符串池中找有沒有”abc”,有的話就把s指向”abc”,沒有就會先在池中創建一個,再讓s指向”abc”。
String s = new String(“abc”);
這句是先定義一個String變量s,然後在內存中分配新的空間”abc”再讓s指向這個內存地址。
它們的實現機制不同,這個和有沒有人接收沒有關係,concat()返回是一個指向常量池111333的全新對象,沒有改變s2的引用。至於replace()方法也有多種實現方式,但據我猜測它的實現就像C中的動態數組一樣,總之不管它是怎麼實現的,sb2最終就是指向了aaaccc。
再多說一句,這個應該在邏輯上理解,而底層實現多種多樣,怎麼實現都不可以,無非就是一個效率問題,concate方法意在返回一個連接之後的數組,而非銷毀對象S2,append方法是附加的意思,意義不言自明了,
原創文章,作者:DJQ79,如若轉載,請註明出處:https://www.506064.com/zh-hant/n/126791.html