多线程并发编程如何正确的执行程序:
- 原子性:执行过程要么成功要么失败,比如经典的银行转账问题。
- 可见性:多线程并发时,一个线程修改了工作内存中的值(主存中的值),会立刻改变主存相应地址的值,其它线程工作内存的值无效,重新获取主存的值。
- 有序性:程序执行的顺序,单个线程中没有依赖的代码,cpu会进行指令重排,使代码执行顺序调换,但是不影响最终执行的结果(单线程没有任何问题,多线程就会出现问题)
总结:解决多线程并发问题,需要程序满足上面三个条件才能正确执行。
volatile能保证第二点可见性。
volatile能禁止指令重排序(所以volatile能在一定程度上保证有序性),但是这里只能保证volatile所修饰的变量之前的程序不会在该变量之后执行,该变量之后的代码不会在变量之前执行。
/*
* 一、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;
}
}