我们本节来了解一个新的解决线程安全性问题的一个方案,我们知道,通过synchronized可以完全的来解决线程中所遇到的安全性问题,那么,我们也知道,synchronized属于一个相对来讲比较重量级的锁,虽然在JDK6的时候已经增加了偏向锁以及轻量级锁,但是,在很多情况下,使用它依然会显得比较笨重,那么,我们本节就来了解一个新的解决线程安全性问题的一个方案,叫做volatile关键字,这么说吧,volatile称之为轻量级锁,被volatile修饰的变量在线程之间是可见的。什么意思呢?首先解释可见的,什么是可见的?就是说,一个线程修改了这个变量的值,在另外一个线程中能够读到这个修改后的值,这就是所谓的可见性,那么,我们之前所说的synchronized除了能让线程之间互斥以外,还有一个非常大的功能,就是保证变量的可见性。什么意思呢?可能大家还是不太理解
那么,咱来举一个例子。
我们使用synchronized来做一件事情,
那么,当多线程的情况下,比如说一个线程在setA,另一个线程在getA,的时候,其实很多情况下会出现线程安全性问题,比如说调用了setA()方法,我们本来是想先让它设置a的值,然后在读a的值,当前a的值是1,那么,一个线程先调用setA(),那么,这个线程还没有执行this.a = a;这段代码的时候,那么,另一个线程就已经调用了getA(),那么,这个时候拿到的a的值就是1,这跟我们所希望的,所预期的结果是不一致的,那么,我们想解决这个问题该怎么办呢?就得加synchronized,保证线程的安全。
那么,加了synchronized之后,我们会发现,当一个线程在进行setA的时候,那么,它先拿到了这把对象锁
然后去给a赋值
那么,当另外一个线程在getA的时候,我们知道,方法上的synchronized锁的是当前类的实例,也就是说,此时synchronized锁已经被第一个线程拿到了,第二个线程就需要等待setA方法执行完毕之后,第二个线程才能够执行getA方法,这样也就保证了a在线程之间的可见性。那么,在这里还是给大家写明白吧,一个main方法,两个线程。
保证可见性的前提是,多个线程必须拿的是同一把锁。