停止不了的异步死循环
同步死循环,可以通过实现Runnable接口来解决死循环,但是在64bit JVM还是会出现死循环,所以还是推荐使用volatile关键字解决,同步死循环,介绍volatile之前,做一个异步死循环的例子
package com.myThread;
public class Thread1 extends Thread {
private boolean isRunning = true;
public boolean isRunning() {
return isRunning;
}
public void setRunning(boolean isRunning) {
this.isRunning = isRunning;
}
@Override
public void run() {
super.run();
System.out.println("run begin");
while (isRunning) {
}
System.out.println("run end");
}
}
package com.test;
import com.myThread.Thread1;
public class Test1 {
public static void main(String[] args) throws InterruptedException {
Thread1 thread1 = new Thread1();
thread1.start();
Thread.sleep(1000);
thread1.setRunning(false);
System.out.println("main evalute");
}
}
打印结果
run begin
main evalute
分析:
没有打印”run end“,关键是因为isRunning这个变量,线程工作访问的是私有堆栈,而thread1.setRunning访问的是共有堆栈。要解决这个问题,可以使用volatile关键字。
volatile关键字
将上面例子的private boolean isRunning = true;
改为volatile private boolean isRunning = true;
打印结果
run begin
main evalute
run end
分析:多线程中有主内存和工作内存之分,在JVM中有一个主内存(公共的),专门负责所有线程共享数据,而每个线程都有自己的私有工作内存(参考书籍 《深入理解java虚拟机》),而volatile访问的是主内存的变量
volatile和synchronized比较
比较 | volatile | synchronized |
---|---|---|
修饰范围 | 变量 | 方法或者代码块 |
多线程访问 | 不阻塞 | 阻塞 |
可见性 | 直接保证 | 间接保证 |
原子性 | 不保证 | 保证 |
作用范围 | 解决变量在多线程间的可见性 | 解决多线程之间访问资源的同步性 |
volatile非原子性
关键字volatile主要使用的场合是在多个线程中可以感知实例变量被更改了,并且可以获得最新的值使用。也就是说volatile不具备同步性,所以也就不具备原子性。
变量在内存工作的过程
当i改变并且使用线程使用volatile,则volatile只保证read 获得最新的值,load、use和assign无法保证原子性,所以多个线程访问同一个实例变量还是要加同步锁