以下内容转自http://ifeve.com/thread-safety-and-immutability/

当多个线程同时访问同一个资源,并且其中的一个或者多个线程对这个资源进行了写操作,才会产生竞态条件。多个线程同时读同一个资源不会产生竞态条件。

我们可以通过创建不可变的共享对象来保证对象在线程间共享时不会被修改,从而实现线程安全。如下示例:

public class ImmutableValue{
private int value = 0; public ImmutableValue(int value){
this.value = value;
} public int getValue(){
return this.value;
}
}

请注意ImmutableValue类的成员变量value是通过构造函数赋值的,并且在类中没有set方法。这意味着一旦ImmutableValue实例被创建,value变量就不能再被修改,这就是不可变性。但你可以通过getValue()方法读取这个变量的值。

(译者注:注意,“不变”(Immutable)和“只读”(Read Only)是不同的。当一个变量是“只读”时,变量的值不能直接改变,但是可以在其它变量发生改变的时候发生改变。比如,一个人的出生年月日是“不变”属性,而一个人的年龄便是“只读”属性,但是不是“不变”属性。随着时间的变化,一个人的年龄会随之发生变化,而一个人的出生年月日则不会变化。这就是“不变”和“只读”的区别。(摘自《Java与模式》第34章))

如果你需要对ImmutableValue类的实例进行操作,可以通过得到value变量后创建一个新的实例来实现,下面是一个对value变量进行加法操作的示例:

public class ImmutableValue{
private int value = 0; public ImmutableValue(int value){
this.value = value;
} public int getValue(){
return this.value;
} public ImmutableValue add(int valueToAdd){
return new ImmutableValue(this.value + valueToAdd);
}
}

请注意add()方法以加法操作的结果作为一个新的ImmutableValue类实例返回,而不是直接对它自己的value变量进行操作。

引用不是线程安全的!

重要的是要记住,即使一个对象是线程安全的不可变对象,指向这个对象的引用也可能不是线程安全的。看这个例子:

public void Calculator{
private ImmutableValue currentValue = null; public ImmutableValue getValue(){
return currentValue;
} public void setValue(ImmutableValue newValue){
this.currentValue = newValue;
} public void add(int newValue){
this.currentValue = this.currentValue.add(newValue);
}
}

Calculator类持有一个指向ImmutableValue实例的引用。注意,通过setValue()方法和add()方法可能会改变这个引用。因此,即使Calculator类内部使用了一个不可变对象,但Calculator类本身还是可变的,因此Calculator类不是线程安全的。换句话说:ImmutableValue类是线程安全的,但使用它的类不是。当尝试通过不可变性去获得线程安全时,这点是需要牢记的。

要使Calculator类实现线程安全,将getValue()、setValue()和add()方法都声明为同步方法即可(下一节讲解)。

最新文章

  1. iOS之ToolBar定制
  2. ehcache2拾遗之cache持久化
  3. 北京地铁站点遍历最少经站次数问题普遍意义上是一个NP问题,目前不存在多项式时间算法能够解决该问题
  4. iOS -Swift 3.0 -UIButton属性大全
  5. JSBinding + SharpKit / 常见问题
  6. MVC文件上传-使用jQuery.FileUpload和Backload组件实现文件上传
  7. ceph源码之一
  8. 老李推荐: 第1章1节《MonkeyRunner源码剖析》概述:前言
  9. Elasticsearch-深入理解索引原理
  10. 修改GeoJson的网址
  11. 【原】javascript笔记之this用法
  12. 1.常用turtle功能函数
  13. Java——IO流 对象的序列化和反序列化流ObjectOutputStream和ObjectInputStream
  14. MT【306】圆与椭圆公切线段
  15. Django——ModuleNotFoundError: No module named 'asgiref.sync'
  16. oracle addm报告
  17. Centos下部署DRBD+NFS+Keepalived高可用环境记录
  18. kettle实现简单的增量同步
  19. iOS 画虚线
  20. 求两个有序数组的中位数或者第k小元素

热门文章

  1. 构造+暴力 Codeforces Round #283 (Div. 2) B. Secret Combination
  2. 题解报告:poj 1426 Find The Multiple(bfs、dfs)
  3. Android 性能优化(24)*性能工具之「Traceview,dmtracedump」Profiling with Traceview and dmtracedump :记录并查看函数调用栈*
  4. canvas学写一个字
  5. System.AppDomain类详解(一)
  6. Set,Map与Array,Object对比
  7. 关于defineProperty
  8. JAVA 学习笔记 - 基础语法 2
  9. ERwin 正向工程
  10. 服务器主机&软件性能测试自定标准