并发编程核心——八、volatile 关键字

8.1、volatile 关键字的概念

自 Java1.5 版本起, volatile 关鍵字所扮演的角色越来越重要,该关鍵字也成为并 发包的基础,所有的原子数据类型都以此作为修饰,相比 synchronized 关键字,volatile 被称为“轻量级锁”,能实现部分 synchronized 关键字的语义。

volatile 概念: volatile 关键字的主要作用是使变量在多个线程间可见。下面我们 通过一个案例来了解一下 volatile:

代码1:未加volatile关键字,一个线程读取变量为true死循环,main线程设置变量为false

package com.bjsxt.chapter08;

import java.util.concurrent.TimeUnit;

public class VolatileDemo extends Thread {
   
    private boolean isRunning = true;// 注意 未加 volatile 修饰的普通变量

    @Override
    public void run() {
   
        System.out.println(Thread.currentThread().getName() + " start run...");
        while(isRunning){
   
            //...
        }
        System.out.println(Thread.currentThread().getName() + " end run...");
    }

    public static void main(String[] args) throws InterruptedException {
   
        VolatileDemo vd = new VolatileDemo();
        vd.start();

        TimeUnit.SECONDS.sleep(1);
        vd.isRunning = false;
        System.out.println("main thread set isRunning false");
    }
}

运行结果:代码处于死循环
分析:main线程获取的是自己工作区的isRunning,赋值后写回到所有线程共享变量中。但是,线程Thread-0使用的是自己工作区的isRunning:true。没有装入步骤。
装入:从共享内存拿数据放进Thread-0线程工作内存
写回:将Thread-0线程工作内存中数据写回共享内存
在这里插入图片描述
代码2:循环体中加入System.out.print…

package com.bjsxt.chapter08;

import java.util.concurrent.TimeUnit;

public class VolatileDemo001 extends Thread  {
   
    private boolean isRunning = true;// 注意 未加 volatile 修饰的普通变量

    @Override
    public void run() {
   
        System.out.println(Thread.currentThread().getName() + " start run...");
        while(isRunning){
   
            //System.out.print("");
            //System.out.printf("");
            System.out.println("");// 以上输出会导致当前线程重新装入
        }
        System.out.println(Thread.currentThread().getName() + " end run...");
    }

    public static void main(String[] args) throws InterruptedException {
   
        VolatileDemo001 vd = new VolatileDemo001();
        vd.start();

        TimeUnit.SECONDS.sleep(1);
        vd.isRunning = false;
        System.out.println("main thread set isRunning false");
    }
}

运行结果:Thread-0的工作内存数据是false。程序不再死循环。
分析:System.out.print…方法会使得 从共享内存拿数据放进Thread-0线程工作内存,导致程序不在死循环。
在这里插入图片描述
代码3:使用 synchronized 解决死循环问题

package com.bjsxt.chapter08;

import java.util.concurrent.TimeUnit;

public class VolatileDemo03 extends Thread {
   
    private boolean isRunning = true;// 注意 未加 volatile 修饰的普通变量

    public synchronized void setRunning(boolean x){
   
        this.isRunning = x;
    }
    public synchronized boolean getRunning(){
   
        return this.isRunning;
    }
    @Override
    public void run() {
   
        System.out.println(Thread.currentThread().getName() + " start run...");
        while(getRunning()){
   
            //...
        }
        System.out.println(Thread.currentThread().getName() + " end run...");
    }

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

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

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

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值