记得刚毕业面试的时候,被问到单例模式的时候,要求写一个线程安全的单例模式出来,当时只知道加synchronized关键字,后来才知道还有volatile这个关键字。
通常听到线程安全、线程同步的时候就要联想到volatile和synchronized关键字,对于并发编程来说基本都会用到,所以了解其中的区别对于之后开发很有意义。
synchronized关键字,它会阻止其它线程获取当前对象的监控锁,这样就使得当前对象中被synchronized关键字保护的代码块无法被其它线程访问,也就无法并发执行。更重要的是,synchronized还会创建一个内存屏障(又称内存栅栏,是一组处理器指令,用于实现对内存操作的顺序限制。),内存屏障指令保证了所有CPU操作结果都会直接刷到主存中,从而保证了操作的内存可见性,同时也使得先获得这个锁的线程的所有操作,都happens-before于随后获得这个锁的线程的操作。
可见性:会使得所有对volatile变量的读写都会直接刷到主存,即保证了变量的可见性。
有序性:volatile可以禁止进行指令重排。
使用volatile关键字仅能实现对原始变量(如boolen、 short 、int 、long等)操作的原子性,但需要特别注意, volatile不能保证复合操作的原子性,即使只是i++,实际上也是由多个原子操作组成:read i; inc; write i,假如多个线程同时执行i++,volatile只能保证他们操作的i是同一块内存,但依然可能出现写入脏数据的情况。
总结:
- volatile本质是在告诉jvm当前变量在寄存器(工作内存)中的值是不确定的,需要从主存中读取; synchronized则是锁定当前变量,只有当前线程可以访问该变量,其他线程被阻塞住。
- volatile仅能使用在变量级别;synchronized则可以使用在变量、方法、和类级别的
- volatile仅能实现变量的修改可见性,不能保证原子性;而synchronized则可以保证变量的修改可见性和原子性
- volatile不会造成线程的阻塞;synchronized可能会造成线程的阻塞。
- volatile标记的变量不会被编译器优化;synchronized标记的变量可以被编译器优化
参考:https://blog.youkuaiyun.com/suifeng3051/article/details/52611233
https://blog.youkuaiyun.com/luoyoub/article/details/80233613