Java 多线程之对象
1. 发布对象
发布对象:使一个对象能够被当前范围之外的代码s所使用
对象溢出:一种错误的对象发布。即:当一个对象还没有构造完成时,就使它被其他的线程所见
(1)发布对象 Demo:(非线程安全)
package object;
import java.util.Arrays;
public class PublishDemo1 {
private String[] test = {"a","b","c"};
public String[] getTest() {
return test;
}
public static void main(String[] args) {
PublishDemo1 demo1 = new PublishDemo1();
System.out.println(Arrays.toString(demo1.getTest()));
demo1.getTest()[0] = "d";
System.out.println(Arrays.toString(demo1.getTest()));
}
}
在此测试代码中,PublishDemo1 通过 public
发布了该类的域。因此,任何外部线程都可以访问这个域
同时通过 new PublishDemo1() 创建了该类的实例,该实例可以直接通过 public 方法访问私有域 test ,此时可以在任何其他线程中修改该值,这样导致不能确定该类的状态是否是确定的。这样的发布对象的方式既是线程不安全的
(2)对象溢出 Demo:(在对象未完成构造之前,不能将其发布,同时线程不安全)
package object;
public class EscapeDemo {
private int sum = 0;
public EscapeDemo() {
new InnerClass();
}
private class InnerClass{
public InnerClass() {
System.out.println(EscapeDemo.this.sum);
}
}
public static void main(String[] args) {
new EscapeDemo();
}
}
在此测试代码中,内部类的实例中包含了对 sum 的隐含引用。即,在对象没有被正确构造之前就进行了发布,导致 this 引用在构造期间就产生了溢出
2. 安全发布对象
安全发布对象实现方式:
- 在静态初始化函数中初始化一个对象的引用
- 将对象的引用保存到 volatile 类型域或者 AtomicReference 对象中