标签:解决 这一 异常 实践 疑问 锁定 public 出现 分配
在一男子给对象转账5000元,居然又退还了!和我就站在你面前,你却视而不见!文中,我们学习了线程安全的原子性和可见性,这篇文章就来说说有序性。首先还是来看下概念,有序性就是指代码按照编写顺序执行。
大家可能会有疑问,难道还会出现乱序执行吗?
因为编译器为了程序性能,可能会改变代码中语句的先后顺序,也就是指令重排序。比如:
String name = "wupx";
Integer age = 18;
编译器优化后可能变成:
Integer age = 18;
String name = "wupx";
在上述的情况中,指令重排序对运行结果没有什么影响,但是指令重排序的优化只能保证单线程程序中是线程安全的。
如果在并发环境下,是不能保证有序性的,这就引出了有序性问题:
下面通过一个双重校验获取单例对象的例子,让大家了解下:
public class Singleton {
private static Singleton instance;
public static Singleton getSingleton() {
if (instance == null) {
synchronized (Singleton.class) {
if (instance == null) {
instance = new Singleton();
}
}
}
return instance;
}
}
这段代码的意思大致就是:首先判断 instance 是不是为空,如果为空,则进入同步代码块并进行再次判空操作,不然直接返回 instance。在同步代码块中再次判断 instance 是否为空,若仍然为空,则创建 Singleton 的实例 instance,其中第二次判空是为了避免了在进入同步代码块时间段内有线程已经创建了 Singleton 的实例。
哇,一个完美的单例模式的实现就完成了,其实并不是,因为代码中的 instance = new Singleton(); 这一行代码对应的 CPU 指令是三个:
但是由于编译器做的指令重排序的优化(可以看出不是代码层面的重排序,是指令层面的重排序),这些命令可能会变成:
不要小瞧编译器做的小动作,我们现在来举例分析下,比如:
如果在第 5 步没有执行完之前,Thread-1 获取到了一个未初始化的 instance,如果在这个时间段内访问 instance 变量,就有可能发生空指针异常(NPE)。
为了方便大家理解,我画个图说明下:
如果要解决这个代码中的有序性问题,可以在 instance 的声明中加上 volatile 关键字,volatile 变量规则是 Happens-Before(先行发生原则) 中的一种:对一个变量的写操作先行发生于后面对这个变量的读操作,因此 volatile 修饰的变量是会保证读操作一定能读到写完的值。
关于 volatile 相关知识,建议阅读:你真的了解 volatile 关键字吗?
在这里再简单介绍下 Happens-Before 规则,Happens-Before 限制了编译器的优化行为,就是要求编译器优化后一定遵守 Happens-Before 规则,我的个人理解就是先前的操作的结果对之后的操作是可见的。
Happens-Before 包括如下规则:
这篇文章,我们一起学习了有序性,并了解了在并发环境下编译器指令重排序优化带来的有序性问题,并在最后简单介绍了 Happens-Before 原则。
到此为止,可见性、原子性、有序性就全部讲解完了,欢迎大家留言讨论,分享你的想法。
最好的关系就是互相成就,大家的在看、转发、留言三连就是我创作的最大动力。
参考
《深入理解Java虚拟机:JVM高级特性与最佳实践》
《实战Java高并发程序设计》
Java并发编程实战
完
●给学妹的Java学习路线
●我就站在你面前,你却视而不见!
●你真的了解 volatile 关键字吗?
武培轩
有帮助?在看,转发走一波
喜欢作者
标签:解决 这一 异常 实践 疑问 锁定 public 出现 分配
原文地址:https://blog.51cto.com/14901336/2522388