您的位置:首页 > 其它

2.7volatile关键字

2016-01-18 09:56 183 查看

停止不了的异步死循环

同步死循环,可以通过实现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 evaluterun end



分析:多线程中有主内存和工作内存之分,在JVM中有一个主内存(公共的),专门负责所有线程共享数据,而每个线程都有自己的私有工作内存(参考书籍 《深入理解java虚拟机》),而volatile访问的是主内存的变量

volatile和synchronized比较

比较volatilesynchronized
修饰范围变量方法或者代码块
多线程访问不阻塞阻塞
可见性直接保证间接保证
原子性不保证保证
作用范围解决变量在多线程间的可见性解决多线程之间访问资源的同步性

volatile非原子性

关键字volatile主要使用的场合是在多个线程中可以感知实例变量被更改了,并且可以获得最新的值使用。也就是说volatile不具备同步性,所以也就不具备原子性。

变量在内存工作的过程



当i改变并且使用线程使用volatile,则volatile只保证read 获得最新的值,load、use和assign无法保证原子性,所以多个线程访问同一个实例变量还是要加同步锁
内容来自用户分享和网络整理,不保证内容的准确性,如有侵权内容,可联系管理员处理 点击这里给我发消息
标签: