2.7volatile关键字

停止不了的异步死循环

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

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

volatile非原子性

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

变量在内存工作的过程

这里写图片描述

当i改变并且使用线程使用volatile,则volatile只保证read 获得最新的值,load、use和assign无法保证原子性,所以多个线程访问同一个实例变量还是要加同步锁

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值