0.概念理解
对象状态 :存储在状态变量(例如实例或静态域)中的数据;
线程安全性 :当多个线程访问某个类时,这个类始终都能表现出正确的行为,那么就称这个类是线程安全的;
竞态条件 :针对一段有多个操作逻辑的代码,不恰当的线程执行时序可能导致不正确的结果的情况;
不变性条件:涉及多个变量时,各个变量之间并不是彼此独立的,而是某个变量的值会对其他变量的值产生约束。
1.什么情况下会面临线程安全性问题?
多个线程需要访问同一组数据,以达到多线程协作的目的时。
2.线程安全性涉及哪些问题?
原子性: 非原子性的操作序列执行结果可能因线程调度执行的顺序不同而得到不一致的结果;
可见性: A线程写入的数据可能不能立即被B线程正确地读取到;
有序性: 程序执行时可能因指令重排并不严格按照代码编写顺序执行(只保证单线程下多次执行结果一致)。
3.要保证线程安全性有哪些工具可用?
sychronized(可见性+原子性),volatile(可见性),显式锁(可见性+原子性),原子变量(可见性+原子性)。
其他需要注意的问题
- 完全由线程安全类构成的程序不一定是线程安全的,线程安全类中也可以包含非线程安全的类;
- 只有当类中仅包含自己的状态时,线程安全类才是有意义的;
- 无状态对象一定是线程安全的;
- 常见的竞态条件是先检查后执行,通过一个可能失效的观测结果来决定下一步的动作;
- 要保持状态一致性,就需要在单个原子操作中更新所有相关的状态变量;
- 内置的sychronized锁是可重入的,场景:子类重写父类的sychronized方法并调用父类的方法;
- 在不变性条件中的每个变量都必须由同一个锁来保护;
- 多个原子性操作的组合使用需要额外加锁;
- 同步代码的执行时间应当尽可能短,把耗时长的代码剥离出去。