本系列译自jakob jenkov的Java并发多线程教程,个人觉得很有收获。由于个人水平有限,不对之处还望矫正!
只有在多个线程访问相同的资源时,才会出现竞态条件,并且一个或多个线对相同的资源进操作。如果多个线程读取相同的资源条件,就不会发生这种情况。
我们通过使共享变量不可以变来确保共享变量不被别的线程修改,因此这样的共享变量是线程安全的,下面有个例子:
public class ImmutableValue{
private int value = 0;
public ImmutableValue(int value){
this.value = value;
}
public int getValue(){
return this.value;
}
}
注意,ImmutableValue的实例是通过构造方法传值的,同时也要注意,这里没有setter方法。因此,只要一个ImmutableValue实例被创建,那么他的值是不可改变的。你可以通过getValue()方法来获取值 。
如果你需要在ImmutableValue实例上执行操作。那么可以通过返回一个带有操作结果的新实例来完成该操作。下面是添加操作的一个示例:
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()方法如何使用add的结果返回一个新的ImmutableValue实例,而不是将此值添加到他自己上。
引用不是线程安全的
重要的是要记住,既使一个不可变的对象是线程安全的。但是如果一旦这个对象被引用,则有可能不是线程安全的。看下面的例子:
public class 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类在内部使用一个不可变的对象,它本身也不是不可变的,因此也不是线程安全的。换句话说:ImmutableValue类是线程安全的,但它的使用却不是。当试图通过不可变性来实现线程安全时,需要牢记这一点。
为了使得Calculator类是线程安全的,你可以声明getValue()、setValue()、add()方法为同步方法,即可做到线程安全。