要想知道volatile和synchronized的区别首先得了解他们并知道他们各自的特点。
synchronized关键字
方式一:同步代码块
synchronized(对象锁){//常用锁:this和类名.class
}
方式二:同步方法,默认锁为this
方式三:同步静态方法,默认锁为类名.class
同步注意事项:
(1)代码尽量简短
(2)不要阻塞
(3)“死锁”: 在持有锁的时候,不要对其它对象调用方法
锁之间不要互相调用
volatile关键字
1.作用:保证被volatile修饰的共享变量 的可见性 & 有序性,但不保证原子性
2.弥补volatile关键字的原子性
(1)理解:synchronized 是不太可控的锁,理解成隐式锁
(2)在JDK1.5以后,出现了Lock,理解成显示锁
(3)在java 1.5的java.util.concurrent.atomic包下提供了一些原子操作类,即对基本数据类型的 自增(加1操作),自减(减1操作)、以及加法操作(加一个数),减法操作(减一个数)进行了封装,保证这些操作是原子性操作。
3.使用:通常是初始化工作,进行读取多,写入少的操作
synchronized的特点:
synchronized关键字解决的是执行控制的问题,它会阻止其它线程获取当前对象的监控锁,这样就使得当前对象中被synchronized关键字保护的代码块无法被其它线程访问,也就无法并发执行。更重要的是,synchronized还会创建一个内存屏障,内存屏障指令保证了所有CPU操作结果都会直接刷到主存中,从而保证了操作的内存可见性,同时也使得先获得这个锁的线程的所有操作,都happens-before于随后获得这个锁的线程的操作。
volatile的特点:
volatile关键字解决的是内存可见性的问题,会使得所有对volatile变量的读写都会直接刷到主存,即保证了变量的可见性。这样就能满足一些对变量可见性有要求而对读取顺序没有要求的需求。
使用volatile关键字仅能实现对原始变量(如boolen、 short 、int 、long等)操作的原子性,但需要特别注意, volatile不能保证复合操作的原子性,即使只是i++,实际上也是由多个原子操作组成:read i; inc; write i,假如多个线程同时执行i++,volatile只能保证他们操作的i是同一块内存,但依然可能出现写入脏数据的情况。
volatile和synchronized的区别
volatile本质是在告诉jvm当前变量在寄存器(工作内存)中的值是不确定的,需要从主存中读取; synchronized则是锁定当前变量,只有当前线程可以访问该变量,其他线程被阻塞住。
volatile仅能使用在变量级别;synchronized则可以使用在变量、方法、和类级别的
volatile仅能实现变量的修改可见性,不能保证原子性;而synchronized则可以保证变量的修改可见性和原子性
volatile不会造成线程的阻塞;synchronized可能会造成线程的阻塞。
volatile标记的变量不会被编译器优化;synchronized标记的变量可以被编译器优化
总结:
简单的来说volatile关键字轻量锁。synchronized和Lock是重量锁。volatile可以来保证可见性。