这两个关键字都是java提供的为保证线程安全而存在的
简单来说,他们的区别
1.synchronized可以保证原子性,以及可见性,而volatile只能保证可见性,也因此性能上volatile更胜一筹
2.volatile不会造成线程阻塞,因为他只是告诉jvm每次使用变量都要获取最新的值,而synchronized则是锁定该变量
3.volatile只能用在变量,synchronized可以用在变量以及方法
具体如何选用还需依照不同场景,个人经验在原始操作上可以使用volatile,入boolean,int,long等,其他符合操作一般使用synchronized
说完他们的区别呢,下面解释下如何产生的这种差异,首先我们要知道java内存模型,如图
每一个线程都有自己的一块本地内存,当需要用某一个变量的时候,会从主内存中拿出这个变量放在本地内存然后再对其进行修改
volatile的作用就是在每次使用变量的时候都从主内存中获取,但是获取完了主内存数据进行刷新,之前获取数据的线程是感知不到的,它再往后使用的数据就不准确了,最常用的例子就是100个线程跑i++,即便获取i的值的时候是准确的,但是执行i++的时候,其他线程可能已经改变了主内存中的值,所以导致数据错误
因此符合操作,即便是i++,都请使用synchronized