1.对象的发布与逸出
发布:使对象能够在当前作用域外的代码中使用
发布的方式:1.通过public属性、方法发布对象
逸出:当某个不应该发布的对象被发布
EventListener隐式地使this引用逸出
public ThisEscape(EventSource source) {
source.registerListener(new EventListener() {
public void onEvent(Event e) {
doSomething(e);
}
});
}
安全的对象构造过程
private final EventListener listener;
private SafeListener() {
listener = new EventListener() {
public void onEvent(Event e) {
doSomething(e);
}
};
}
public static SafeListener newInstance(EventSource source) {
SafeListener safe = new SafeListener();
source.registerListener(safe.listener);
return safe;
}
void doSomething(Event e) {
对象未初始化完成就发生逸出,会出现不可预知的错误
2.线程封闭
当访问共享的可变数据时,通常需要同步,避免使用同步的方式,不共享数据,线程内部访问数据,线程封闭技术
Ad-hoc线程封闭:维护线程封闭性的职责完全由程序实现来承担
栈封闭:由于任何方法都无法获取基本类型的引用,确保基本类型的局部变量始终封闭在线程中。
ThreadLocal类:ThreadLocal提供get与set方法,每个线程都有独立的副本,因此get总能获取到最新值
3.不变性
不可变的对象一定是线程安全的
当满足一下条件时,对象才是不可变的
对象创建以后状态不可修改
对象的所有域都是final类型
对象是正确创建(初始化,不存在this逸出)
4.Final域
final类型的域是不能修改的,可能大家会迷惑,熟悉JAVA运行内存你一定会明白,final所修饰的不可变是指的是栈上的引用不可变,引用的所指向的对象是可变的
5.完全发布(常用模式)
1.静态初始化函数中初始化一个对象的引用
2.用volatite类型或者AtomicReferance对象中
3.将对象的引用保存到某个正确构造对象的final类型域中
4.保存到一个锁保护的域中
5.安全容器类中发布
6.future和Exchanger同样能安全发布
6.事实不可变对象(从技术上将是可变的,但其状态不会再改变),可以提高性能
7.对象的发布取决于它的可变性
不可变对象可以通过任意机制来发布
事实不可变对象必须通过安全方式来发布
可变对象必须通过安全方式发布,并且必须是线程安全的或者锁来保护
8.安全地共享对象
线程封闭
只读共享
保护对象(加锁)