volatile修饰的变量可以保证全部线程的可见性,即当数据修改后全部线程都可以访问到修改后的数据安全
synchronized同步锁,表示在多线程环境下只能有一个线程进入被synchronized修饰过的代码块、方法、变量、对象。多线程
修饰对象不一样,volatile只能修饰于变量,而synchronized能够修饰代码块、方法、变量、对象线程
volatile保证了可见性和有序性,synchronized保证了原子性对象
每一个线程从本身线程独占的cache取数据,而cache经过JMM取到在主存上的数据,而这些数据通讯由JVM控制,线程间不能直接通讯,何时将数据同步到主存,何时从主存取数据都要靠JVMblog
那么这种模型在多线程模型中保证了安全,可是效率低下编译器
而volatile的原理是将volatile修饰的变量在修改后强制同步到主存,而且每次使用这个数据都要去主存中更新同步
当保证了变量的可见性后,在编译器和CPU级别上都会由于Memory Barrier指令禁止指令重排编译
在jdk1.6以前,synchronized一直都是重量级锁,1.6以后引入了偏向锁、轻量级锁,如今对象就有了四个状态:无锁、偏向锁、轻量级锁、重量级锁效率
每一个对象都有对象头,而对于每一个对象的锁状态都是在对象头上的markword上变量
当线程进入同步区域时,若是同步对象的对象头锁标志位为"01"则表示可进入,进入后将标志位设置为"00"表示这个线程得到了锁,而且将这个对象设置轻量级锁锁定状态,若是这个更新操做失败,而且检测是否有多个线程在竞争当前锁,若是有多个线程竞争则会转换为重量级锁而且使其余线程阻塞,而当前线程则自旋等待。
当线程请求到锁时,将锁对象的标志位改成"01",而后将线程的ID记录到MarkWord中,之后该线程将能够直接进入该同步区域,可是若是有其余线程请求锁,将会当即转换为轻量级锁模式。
无锁、偏向锁、轻量级锁、重量级锁状态切换