国产探花免费观看_亚洲丰满少妇自慰呻吟_97日韩有码在线_资源在线日韩欧美_一区二区精品毛片,辰东完美世界有声小说,欢乐颂第一季,yy玄幻小说排行榜完本

首頁 > 學院 > 開發設計 > 正文

StringBuffer與StringBuilder原理與區別

2019-11-14 15:24:34
字體:
來源:轉載
供稿:網友
其實只要找下Google大神就有答案了:StringBuffer 與 StringBuilder 中的方法和功能完全是等價的,只是StringBuffer 中的方法大都采用了 synchronized 關鍵字進行修飾,因此是線程安全的,而 StringBuilder 沒有這個修飾,可以被認為是線程不安全的。 
 
為了更好的理解上述的答案,還是直接看StringBuffer與StringBuilder的源碼實現比較實在,作為一個程序猿,“有疑問,看源碼”才是正道,我可以負責任的說,當然了得有條件才行!
 
jdk的實現中StringBuffer與StringBuilder都繼承自AbstractStringBuilder,對于多線程的安全與非安全看到StringBuffer中方法前面的一堆synchronized就大概了解了。
 
這里隨便講講AbstractStringBuilder的實現原理:我們知道使用StringBuffer等無非就是為了提高java中字符串連接的效率,因為直接使用+進行字符串連接的話,jvm會創建多個String對象,因此造成一定的開銷。AbstractStringBuilder中采用一個char數組來保存需要append的字符串,char數組有一個初始大小,當append的字符串長度超過當前char數組容量時,則對char數組進行動態擴展,也即重新申請一段更大的內存空間,然后將當前char數組拷貝到新的位置,因為重新分配內存并拷貝的開銷比較大,所以每次重新申請內存空間都是采用申請大于當前需要的內存空間的方式,這里是2倍。
 
 
接下來,玩些好玩的!
在Google中出來了這么一些信息:
 
    StringBuffer 始于 JDK 1.0 
    StringBuilder 始于 JDK 1.5 

    從 JDK 1.5 開始,帶有字符串變量的連接操作(+),JVM 內部采用的是 
    StringBuilder 來實現的,而之前這個操作是采用 StringBuffer 實現的。
 
我們通過一個簡單的程序來看其執行的流程:
 
清單1 Buffer.java
[java] view plaincopy
  1. public class Buffer {  
  2.      public static void main(String[] args) {  
  3.             String s1 = "aaaaa";  
  4.             String s2 = "bbbbb";  
  5.             String r = null;  
  6.             int i = 3694;  
  7.             r = s1 + i + s2;   
  8.               
  9.             for(int j=0;i<10;j++){  
  10.                 r+="23124";  
  11.             }  
  12.      }  
  13. }  
 
使用命令javap -c Buffer查看其字節碼實現:
 
清單2 Buffer類字節碼
 
將清單1和清單2對應起來看,清單2的字節碼中ldc指令即從常量池中加載“aaaaa”字符串到棧頂,istore_1將“aaaaa”存到變量1中,后面的一樣,sipush是將一個短整型常量值(-32768~32767)推送至棧頂,這里是常量“3694”,更多的Java指令集請查看另一篇文章“Java指令集”。
 
讓我們直接看到13,13~17是new了一個StringBuffer對象并調用其初始化方法,20~21則是先通過aload_1將變量1壓到棧頂,前面說過變量1放的就是字符串常量“aaaaa”,接著通過指令invokevirtual調用StringBuffer的append方法將“aaaaa”拼接起來,后續的24~30同理。最后在33調用StringBuffer的toString函數獲得String結果并通過astore存到變量3中。
 
看到這里可能有人會說,“既然JVM內部采用了StringBuffer來連接字符串了,那么我們自己就不用用StringBuffer,直接用”+“就行了吧!“。是么?當然不是了。俗話說”存在既有它的理由”,讓我們繼續看后面的循環對應的字節碼。
 
37~42都是進入for循環前的一些準備工作,37,38是將j置為1。44這里通過if_icmpge將j與10進行比較,如果j大于10則直接跳轉到73,也即return語句退出函數;否則進入循環,也即47~66的字節碼。這里我們只需看47到51就知道為什么我們要在代碼中自己使用StringBuffer來處理字符串的連接了,因為每次執行“+”操作時jvm都要new一個StringBuffer對象來處理字符串的連接,這在涉及很多的字符串連接操作時開銷會很大。

發表評論 共有條評論
用戶名: 密碼:
驗證碼: 匿名發表
主站蜘蛛池模板: 阿克苏市| 邵武市| 敦煌市| 林州市| 津南区| 安仁县| 定兴县| 宿松县| 凤翔县| 迁西县| 江源县| 平和县| 含山县| 监利县| 泽普县| 双牌县| 沾化县| 灌云县| 石景山区| 乐山市| 云霄县| 咸宁市| 彩票| 奉新县| 比如县| 天祝| 仙游县| 日喀则市| 台州市| 金堂县| 天水市| 永泰县| 赣榆县| 富平县| 咸丰县| 琼中| 梁山县| 恩平市| 阳谷县| 余姚市| 贞丰县|