2.7volatile关键字

来源:互联网 发布:济南直销软件 编辑:程序博客网 时间:2024/06/18 04:41

停止不了的异步死循环

同步死循环,可以通过实现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无法保证原子性,所以多个线程访问同一个实例变量还是要加同步锁

0 0
原创粉丝点击