欧美bbbwbbbw肥妇,免费乱码人妻系列日韩,一级黄片

JAVA面試題 從源碼角度分析StringBuffer和StringBuilder的區(qū)別

 更新時(shí)間:2019年07月22日 08:28:19   作者:Java螞蟻  
這篇文章主要介紹了JAVA面試題 從源碼角度分析StringBuffer和StringBuilder的區(qū)別,文中通過示例代碼介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,下面我們來一起學(xué)習(xí)下吧

面試官:請(qǐng)問StringBuffer和StringBuilder有什么區(qū)別?

這是一個(gè)老生常談的話題,筆者前幾年每次面試都會(huì)被問到,作為基礎(chǔ)面試題,被問到的概率百分之八九十。下面我們從面試需要答到的幾個(gè)知識(shí)點(diǎn)來總結(jié)一下兩者的區(qū)別有哪些?

  • 繼承關(guān)系?
  • 如何實(shí)現(xiàn)的擴(kuò)容?
  • 線程安全性?

繼承關(guān)系

從源碼上看看類StringBuffer和StringBuilder的繼承結(jié)構(gòu):

從結(jié)構(gòu)圖上可以直到,StringBuffer和StringBuiler都繼承自AbstractStringBuilder類

如何實(shí)現(xiàn)擴(kuò)容

StringBuffer和StringBuiler的擴(kuò)容的機(jī)制在抽象類AbstractStringBuilder中實(shí)現(xiàn),當(dāng)發(fā)現(xiàn)長度不夠的時(shí)候(默認(rèn)長度是16),會(huì)自動(dòng)進(jìn)行擴(kuò)容工作,擴(kuò)展為原數(shù)組長度的2倍加2,創(chuàng)建一個(gè)新的數(shù)組,并將數(shù)組的數(shù)據(jù)復(fù)制到新數(shù)組。

public void ensureCapacity(int minimumCapacity) {
 if (minimumCapacity > 0)
  ensureCapacityInternal(minimumCapacity);
}
 
/**
* 確保value字符數(shù)組不會(huì)越界.重新new一個(gè)數(shù)組,引用指向value
*/ 
private void ensureCapacityInternal(int minimumCapacity) {
 // overflow-conscious code
 if (minimumCapacity - value.length > 0) {
  value = Arrays.copyOf(value,
    newCapacity(minimumCapacity));
 }
}
 
/**
* 擴(kuò)容:將長度擴(kuò)展到之前大小的2倍+2
*/ 
private int newCapacity(int minCapacity) {
 // overflow-conscious code 擴(kuò)大2倍+2
 //這里可能會(huì)溢出,溢出后是負(fù)數(shù)哈,注意
 int newCapacity = (value.length << 1) + 2;
 if (newCapacity - minCapacity < 0) {
  newCapacity = minCapacity;
 }
 //MAX_ARRAY_SIZE的值是Integer.MAX_VALUE - 8,先判斷一下預(yù)期容量(newCapacity)是否在0<x<MAX_ARRAY_SIZE之間,在這區(qū)間內(nèi)就直接將數(shù)值返回,不在這區(qū)間就去判斷一下是否溢出
 return (newCapacity <= 0 || MAX_ARRAY_SIZE - newCapacity < 0)
  ? hugeCapacity(minCapacity)
  : newCapacity;
}
 
/**
* 判斷大小,是否溢出
*/
private int hugeCapacity(int minCapacity) {
 if (Integer.MAX_VALUE - minCapacity < 0) { // overflow
  throw new OutOfMemoryError();
 }
 return (minCapacity > MAX_ARRAY_SIZE)
  ? minCapacity : MAX_ARRAY_SIZE;
}

線程安全性

我們先來看看StringBuffer的相關(guān)方法:

@Override
public synchronized StringBuffer append(long lng) {
 toStringCache = null;
 super.append(lng);
 return this;
}
 
/**
 * @throws StringIndexOutOfBoundsException {@inheritDoc}
 * @since  1.2
 */
@Override
public synchronized StringBuffer replace(int start, int end, String str) {
 toStringCache = null;
 super.replace(start, end, str);
 return this;
}
 
/**
 * @throws StringIndexOutOfBoundsException {@inheritDoc}
 * @since  1.2
 */
@Override
public synchronized String substring(int start) {
 return substring(start, count);
}
 
@Override
public synchronized String toString() {
 if (toStringCache == null) {
  toStringCache = Arrays.copyOfRange(value, 0, count);
 }
 return new String(toStringCache, true);
}

從上面的源碼中我們看到幾乎都是所有方法都加了synchronized,幾乎都是調(diào)用的父類的方法.,用synchronized關(guān)鍵字修飾意味著什么?加鎖,資源同步串行化處理,所以是線程安全的。

我們?cè)賮砜纯碨tringBuilder的相關(guān)源碼:

@Override
public StringBuilder append(double d) {
 super.append(d);
 return this;
}
 
/**
 * @since 1.5
 */
@Override
public StringBuilder appendCodePoint(int codePoint) {
 super.appendCodePoint(codePoint);
 return this;
}
 
/**
 * @throws StringIndexOutOfBoundsException {@inheritDoc}
 */
@Override
public StringBuilder delete(int start, int end) {
 super.delete(start, end);
 return this;
}

StringBuilder的源碼里面,基本上所有方法都沒有用synchronized關(guān)鍵字修飾,當(dāng)多線程訪問時(shí),就會(huì)出現(xiàn)線程安全性問題。

為了證明StringBuffer線程安全,StringBuilder線程不安全,我們通過一段代碼進(jìn)行驗(yàn)證:

測試思想

  • 分別用1000個(gè)線程寫StringBuffer和StringBuilder,
  • 使用CountDownLatch保證在各自1000個(gè)線程執(zhí)行完之后才打印StringBuffer和StringBuilder長度,
  • 觀察結(jié)果。

測試代碼

import java.util.concurrent.CountDownLatch;
 
public class TestStringBuilderAndStringBuffer {
 public static void main(String[] args) {
  //證明StringBuffer線程安全,StringBuilder線程不安全
  StringBuffer stringBuffer = new StringBuffer();
  StringBuilder stringBuilder = new StringBuilder();
  CountDownLatch latch1 = new CountDownLatch(1000);
  CountDownLatch latch2 = new CountDownLatch(1000);
  for (int i = 0; i < 1000; i++) {
   new Thread(new Runnable() {
    @Override
    public void run() {
     try {
      stringBuilder.append(1);
     } catch (Exception e) {
      e.printStackTrace();
     } finally {
      latch1.countDown();
     }
    }
   }).start();
  }
  for (int i = 0; i < 1000; i++) {
   new Thread(new Runnable() {
    @Override
    public void run() {
     try {
      stringBuffer.append(1);
     } catch (Exception e) {
      e.printStackTrace();
     } finally {
      latch2.countDown();
     }
 
    }
   }).start();
  }
  try {
   latch1.await();
   System.out.println(stringBuilder.length());
   latch2.await();
   System.out.println(stringBuffer.length());
  } catch (InterruptedException e) {
   e.printStackTrace();
  }
 }
}

測試結(jié)果

  • StringBuffer不論運(yùn)行多少次都是1000長度。
  • StringBuilder絕大多數(shù)情況長度都會(huì)小于1000。
  • StringBuffer線程安全,StringBuilder線程不安全得到證明。

總結(jié)一下

  • StringBuffer和StringBuilder都繼承自抽象類AbstractStringBuilder。
  • 存儲(chǔ)數(shù)據(jù)的字符數(shù)組也沒有被final修飾,說明值可以改變,且構(gòu)造出來的字符串還有空余位置拼接字符串,但是拼接下去肯定也有不夠用的時(shí)候,這時(shí)候它們內(nèi)部都提供了一個(gè)自動(dòng)擴(kuò)容機(jī)制,當(dāng)發(fā)現(xiàn)長度不夠的時(shí)候(默認(rèn)長度是16),會(huì)自動(dòng)進(jìn)行擴(kuò)容工作,擴(kuò)展為原數(shù)組長度的2倍加2,創(chuàng)建一個(gè)新的數(shù)組,并將數(shù)組的數(shù)據(jù)復(fù)制到新數(shù)組,所以對(duì)于拼接字符串效率要比String要高。自動(dòng)擴(kuò)容機(jī)制是在抽象類中實(shí)現(xiàn)的。
  • 線程安全性:StringBuffer效率低,線程安全,因?yàn)镾tringBuffer中很多方法都被 synchronized 修飾了,多線程訪問時(shí),線程安全,但是效率低下,因?yàn)樗屑渔i和釋放鎖的過程。StringBuilder效率高,但是線程是不安全的。

以上就是本文的全部內(nèi)容,希望對(duì)大家的學(xué)習(xí)有所幫助,也希望大家多多支持腳本之家。

相關(guān)文章

  • Java回調(diào)函數(shù)與觀察者模式實(shí)例代碼

    Java回調(diào)函數(shù)與觀察者模式實(shí)例代碼

    這篇文章主要介紹了Java回調(diào)函數(shù)與觀察者模式實(shí)例代碼,簡單介紹了使用觀察者模式的場景,分享了相關(guān)代碼示例,小編覺得還是挺不錯(cuò)的,具有一定借鑒價(jià)值,需要的朋友可以參考下
    2018-02-02
  • Java之JsonArray用法講解

    Java之JsonArray用法講解

    這篇文章主要介紹了Java之JsonArray用法講解,本篇文章通過簡要的案例,講解了該項(xiàng)技術(shù)的了解與使用,以下就是詳細(xì)內(nèi)容,需要的朋友可以參考下
    2021-08-08
  • JSP 獲取spring容器中bean的兩種方法總結(jié)

    JSP 獲取spring容器中bean的兩種方法總結(jié)

    這篇文章主要介紹了JSP 獲取spring容器中bean的方法總結(jié)的相關(guān)資料,需要的朋友可以參考下
    2017-05-05
  • Spring使用注解進(jìn)行對(duì)象注入的示例詳解

    Spring使用注解進(jìn)行對(duì)象注入的示例詳解

    獲取?Bean?對(duì)象也叫做對(duì)象裝配,就是把對(duì)象取出來放到某個(gè)類中,有時(shí)候也叫對(duì)象注入,常見有關(guān)對(duì)象注入的注解有兩個(gè),一個(gè)是@Autowired,另外一個(gè)是@Resource,下面就來講講它們的具體使用吧
    2023-07-07
  • java學(xué)習(xí)之利用TCP實(shí)現(xiàn)的簡單聊天示例代碼

    java學(xué)習(xí)之利用TCP實(shí)現(xiàn)的簡單聊天示例代碼

    這篇文章主要給大家介紹了關(guān)于java學(xué)習(xí)筆記之利用TCP實(shí)現(xiàn)的簡單聊天的相關(guān)資料,文中通過示例代碼介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友們下面隨著小編來一起學(xué)習(xí)學(xué)習(xí)吧。
    2017-12-12
  • 詳談Java中的Object、T(泛型)、?區(qū)別

    詳談Java中的Object、T(泛型)、?區(qū)別

    下面小編就為大家?guī)硪黄斦凧ava中的Object、T(泛型)、?區(qū)別。小編覺得挺不錯(cuò)的,現(xiàn)在就分享給大家,也給大家做個(gè)參考。一起跟隨小編過來看看吧
    2017-06-06
  • ShardingSphere-Proxy5搭建使用過程分析

    ShardingSphere-Proxy5搭建使用過程分析

    ShardingSphere-Proxy是跨語言的數(shù)據(jù)庫代理服務(wù)端,主要用來處理:分表、分庫、讀寫分離 等,這篇文章主要介紹了ShardingSphere-Proxy5搭建使用過程,需要的朋友可以參考下
    2022-10-10
  • 解決Lombok注解不起作用的問題

    解決Lombok注解不起作用的問題

    這篇文章主要介紹了解決Lombok注解不起作用的問題,具有很好的參考價(jià)值,希望對(duì)大家有所幫助。如有錯(cuò)誤或未考慮完全的地方,望不吝賜教
    2021-08-08
  • IDEA全局查找關(guān)鍵字的用法解讀

    IDEA全局查找關(guān)鍵字的用法解讀

    這篇文章主要介紹了IDEA全局查找關(guān)鍵字的用法解讀,具有很好的參考價(jià)值,希望對(duì)大家有所幫助。如有錯(cuò)誤或未考慮完全的地方,望不吝賜教
    2023-02-02
  • 詳解Java中的增強(qiáng) for 循環(huán) foreach

    詳解Java中的增強(qiáng) for 循環(huán) foreach

    foreach 是 Java 中的一種語法糖,幾乎每一種語言都有一些這樣的語法糖來方便程序員進(jìn)行開發(fā),編譯期間以特定的字節(jié)碼或特定的方式來對(duì)這些語法進(jìn)行處理。能夠提高性能,并減少代碼出錯(cuò)的幾率。
    2017-05-05

最新評(píng)論