volatile关键字与内存可见性
看如下一段代码
package com.atguigu.juc;
/*
* 一、volatile 关键字:当多个线程进行操作共享数据时,可以保证内存中的数据可见。
* 相较于 synchronized 是一种较为轻量级的同步策略。
*
* 注意:
* 1. volatile 不具备“互斥性”
* 2. volatile 不能保证变量的“原子性”
*/
public class TestVolatile {
public static void main(String[] args) {
ThreadDemo td = new ThreadDemo();
new Thread(td).start();
while(true){
if(td.isFlag()){
System.out.println("------------------");
break;
}
}
}
}
class ThreadDemo implements Runnable {
private volatile boolean flag = false;
@Override
public void run() {
try {
Thread.sleep(200);
} catch (InterruptedException e) {
}
flag = true;
System.out.println("flag=" + isFlag());
}
public boolean isFlag() {
return flag;
}
public void setFlag(boolean flag) {
this.flag = flag;
}
}
如果成员变量flag没有用volatile声明,那么打印不出“-----”。
原因是线程1运行了将flag赋值为true,即使main线程和线程td共享数据flag,但是main线程执行速度很快,在td线程将flag=true返回给主存之前,main线程就已经从主存拿到了flag=false。
内存可见性问题就是,当多个线程操作共享数据时,彼此不可见
方法一:
用同步锁解决
synchronized(td){
if(td.isFlag()){
System.out.println("------------------");
break;
}
}
让线程td重复地到主存中去读数据。
但是当多个线程去访问这段时,如果一个线程正在占用这个锁,其他线程就只能挂起,造成阻塞。效率低
volatile关键字
当多个线程去操作共享数据时候,可以保证内存中的数据可见,相较于synchronized是一种轻量级的同步策略
注意:
1.volatile 不具备“互斥性”
2.volatile 不能保证变量的“原子性”。