为什么不要用 StringBuffer 和 Vector

原文链接

很多人在回答新手提问的时候会说,StringBuilder 是非同步的,所以快一些,StringBuffer是同步(线程安全)的,所以慢一些。

从技术上说这句话是没有错的,但是为什么多线程的时候仍然用 StringBuilder,不建议用StringBuffer(StringBuffer 建议永远不要使用)?

简单的说,StringBuffer的“线程安全”在大多数时候增加了大量不必要的成本,并且未必达到目的。
看下面的代码:

StringBuffer buffer = getStringBuffer();
for(int i=0; i<10000; i++)   buffer.append(i); // append 方法上由关键字 synchronized ,是同步方法

StringBuffer 的每一步操作都先获取同步锁,然后释放同步锁。
所以这段代码中,它获取了10000次锁,释放了10000次锁。

除了获取、释放10000次锁的高昂成本之外,它真的实现了“线程安全”的目的吗?
假如在这个for循环运行的过程中,有另一个线程对 buffer 做了操作:

buffer.append("some string");

那么这个 "some string" 就会插入某两个数字之间!因为在任意两次 append 数字之间,有一个“释放锁”再“获取锁”的极短暂的间隔,这时候是可以被其他线程获取锁的。
换句话说,StringBuffer 所提供的所谓“线程安全”,并不包括多个操作之间的“原子性”支持。

要想前述的for循环不受干扰地完成,用户还是需要手动上锁:

final StringBuffer buffer = getStringBuffer();
synchronized(buffer) {
  for(int i=0; i<10000; i++)
        buffer.append(i);
}

这样是保证了在for循环运行期间不受其他线程干扰,可是如前所述,10000次锁值 +1 和 -1 的成本还是有的,并且这个成本变成了“完全没有必要”的无意义成本。
这就是为什么 StringBuffer这个类天生缺陷。

在API设计中有一条规则是这样说的:

一个类不应该自己实现同步,而应该把同步的工作留给用户。

在绝大多数情况下,把同步的工作留给应用代码,而不是工具类的代码。

这是因为: 一个类的用户总是比它的设计者更清楚,什么时候应该同步,应该作什么样的同步,并且,在不同的环境下,用户可能对“如何同步”有不同的需求.

因为同样的原因,Vector 也有这样的天生设计缺陷,java的设计者认识到这种缺陷,所以后来才有了 ArrayList 才有了 StringBuilder。

不幸的是,在API的编写中有这样一种规则:任何公开的东西都已经被冻结,任何签名、接口、承诺、结构等等都不能再改动,因为公开就表示已经有大量的用户代码依赖于此。用户能看到的所有的东西他们都会去依赖,假如你做改动,就会有成千上万的用户代码不能再运行或不能再编译。
所以这两个类还在哪里,但是不表示我们在新的代码里还要去用它们。

最后总结:
100%的情况下不要用 StringBuffer
99% 的情况下不要用 Vector
那么那剩下的 1% 用 Vector 的情况在哪呢?
熟悉Swing的都知道: 现有的一些 model 的类里面用了 Vector,假如你去定制它们,有时不可避免要用到 Vector。

©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 221,548评论 6 515
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 94,497评论 3 399
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 167,990评论 0 360
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 59,618评论 1 296
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 68,618评论 6 397
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 52,246评论 1 308
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 40,819评论 3 421
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 39,725评论 0 276
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 46,268评论 1 320
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 38,356评论 3 340
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 40,488评论 1 352
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 36,181评论 5 350
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 41,862评论 3 333
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 32,331评论 0 24
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 33,445评论 1 272
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 48,897评论 3 376
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 45,500评论 2 359

推荐阅读更多精彩内容

  • 1. Java基础部分 基础部分的顺序:基本语法,类相关的语法,内部类的语法,继承相关的语法,异常的语法,线程的语...
    子非鱼_t_阅读 31,662评论 18 399
  • (一)Java部分 1、列举出JAVA中6个比较常用的包【天威诚信面试题】 【参考答案】 java.lang;ja...
    独云阅读 7,114评论 0 62
  • 跟朋友喝茶有不知不觉的喝到这个点,一看时间不早了而且今天的写作计划还没有完成,就赶紧结束。现在我正站在楼下打车...
    hey_金燕阅读 205评论 0 0
  • 1. Swift枚举基本概念 Swift中的枚举比OC中的枚举强大, 因为Swift中的枚举是一等类型, 它可以像...
    DwightChan阅读 1,913评论 0 4
  • JavaScript是ECMAScript的实现和扩展,ES6标准的制定也为JavaScript加入了许多新特性。...
    hope7th阅读 5,330评论 0 0