volatile的作用
阅读本篇前参见上一篇博文https://www.cnblogs.com/dearnotes/p/12290564.html
1.保证此变量对所有线程的可见性
(当一条线程修改了这个变量的值,新值对于其他线程来说立即得知)
虽然是这样,但是我们并不能因此认为基于violate变量的运算在并发下是安全的。
public class VolatileTest extends Thread{
static volatile int increase = 0;
static AtomicInteger aInteger=new AtomicInteger();//对照组
static void increaseFun() {
increase++;
aInteger.incrementAndGet();
}
public void run(){
int i=0;
while (i < 10000) {
increaseFun();
i++;
}
}
public static void main(String[] args) {
VolatileTest vt = new VolatileTest();
int THREAD_NUM = 10;
Thread[] threads = new Thread[THREAD_NUM];
for (int i = 0; i < THREAD_NUM; i++) {
threads[i] = new Thread(vt, "线程" + i);
threads[i].start();
}
//idea中会返回主线程和守护线程,如果用Eclipse的话改为1
while (Thread.activeCount() > 2) {
Thread.yield();
}
System.out.println("volatile的值: "+increase);
System.out.println("AtomicInteger的值: "+aInteger);
}
}
这段代码如果能保证并发安全的话结果应该是100000,但是实际结果会小于这个数字。
为什么呢?
volatile修饰的变量不能保证它的原子性。
由于volate变量只能保证可见性,在不符合以下两条运算场景中,我们仍然要通过加锁(使用synchronized或java.util.concurrent中的原子类)来保证原子性。
- 运算结果并不依赖变量的当前值,或者能够确保只有单一的线程修改变量的值
- 变量不需要与其他的状态变量共同参与不变约束
单独的violate适用场景参见<a href="https://blog.csdn.net/vking_wang/article/details/9982709" target=_blank">https://blog.csdn.net/vking_wang/article/details/9982709