JMM及Volatile

了解JMM需要我们去知道什么是Volatile。

谈谈你对Volatile的理解:
1.保证可见性
2.不保证原子性
3.禁止指令重排

那么什么是JMM呢?其实JMM只是一个概念,并不是显示存在的东西。
JMM:Java内存模型

关于JMM的同步约定:
1.线程解锁前需要共享变量立刻刷到主存中。
2.线程加锁前必须读取主存中的最新值到工作内存中。
3.加锁和解锁必须是一把锁。

内存划分

JMM的内存主要划分为主内存和工作内存
在这里插入图片描述

内存相关操作

内存交互操作有8种,虚拟机实现必须保证每一个操作都是原子的,不可在分的(对于double和long类型的变量来说,load、store、read和write操作在某些平台上允许例外)

  • lock (锁定):作用于主内存的变量,把一个变量标识为线程独占状态
  • unlock (解锁):作用于主内存的变量,它把一个处于锁定状态的变量释放出来,释放后的变量才可以被其他线程锁定
  • read (读取):作用于主内存变量,它把一个变量的值从主内存传输到线程的工作内存中,以便随后的load动作使用
  • load (载入):作用于工作内存的变量,它把read操作从主存中变量放入工作内存中
  • use (使用):作用于工作内存中的变量,它把工作内存中的变量传输给执行引擎,每当虚拟机遇到一个需要使用到变量的值,就会使用到这个指令
  • assign (赋值):作用于工作内存中的变量,它把一个从执行引擎中接受到的值放入工作内存的变量副本中
  • store (存储):作用于主内存中的变量,它把一个从工作内存中一个变量的值传送到主内存中,以便后续的write使用
  • write  (写入):作用于主内存中的变量,它把store操作从工作内存中得到的变量的值放入主内存的变量中

JMM对这八种指令的使用,制定了如下规则:

  • 不允许read和load、store和write操作之一单独出现。即使用了read必须load,使用了store必须write
  • 不允许线程丢弃他最近的assign操作,即工作变量的数据改变了之后,必须告知主存
  • 不允许一个线程将没有assign的数据从工作内存同步回主内存
  • 一个新的变量必须在主内存中诞生,不允许工作内存直接使用一个未被初始化的变量。就是怼变量实施use、store操作之前,必须经过assign和load操作
  • 一个变量同一时间只有一个线程能对其进行lock。多次lock后,必须执行相同次数的unlock才能解锁
  • 如果对一个变量进行lock操作,会清空所有工作内存中此变量的值,在执行引擎使用这个变量前,必须重新load或assign操作初始化变量的值
  • 如果一个变量没有被lock,就不能对其进行unlock操作。也不能unlock一个被其他线程锁住的变量
  • 对一个变量进行unlock操作之前,必须把此变量同步回主内存

刚刚谈到了Volatile,Volatile不保证原子性,那么我们怎么才能让它保证原子性呢?
方法一:加锁 Synchronized或者Lock锁。
方法二:JDK为我们提供了一个保证原子性的包:
在这里插入图片描述
‘使用了Volatile

package com.ys.VTest;


public class VaTest {
    private static volatile int num = 0;

    public  static void add(){
        num++;
    }

    public static void main(String[] args) {

        for (int i = 0; i < 10; i++) {
            new Thread(()->{
                for (int i1 = 0; i1 < 100; i1++) {
                    add();
                }
            }).start();
        }

        while (Thread.activeCount()>2){
            Thread.yield();  //线程礼让
        }



        System.out.println(num);


    }


}


结果是不保证原子性的,每次的结果都不经相同,那么怎么解决呢?
在add方法上锁就能解决(以Synchronized为例):

package com.ys.VTest;


public class VaTest {
    private static volatile int num = 0;

    public synchronized   static void add(){
        num++;
    }

    public static void main(String[] args) {

        for (int i = 0; i < 10; i++) {
            new Thread(()->{
                for (int i1 = 0; i1 < 100; i1++) {
                    add();
                }
            }).start();
        }

        while (Thread.activeCount()>2){
            Thread.yield();
        }



        System.out.println(num);


    }


}

结果:
在这里插入图片描述
现在有个问题就是我们要保证原子性但是不用加锁的方式去解决,应该怎么办?
使用原子类
对以上代码修改:

package com.ys.VTest;


import java.util.concurrent.atomic.AtomicLong;

public class VaTest {
    private static volatile AtomicLong num = new AtomicLong();

    public  static void add(){
        num.getAndIncrement();//加1
    }

    public static void main(String[] args) {

        for (int i = 0; i < 10; i++) {
            new Thread(()->{
                for (int i1 = 0; i1 < 100; i1++) {
                    add();
                }
            }).start();
        }

        while (Thread.activeCount()>2){
            Thread.yield();
        }



        System.out.println(num);


    }


}

在这里插入图片描述

原子类的包装类是CAS,原子类比较高效,比锁还要高效。

以上是不保证原子性的,接下来验证下保证可见性。

package com.ys.VTest;

import java.util.concurrent.TimeUnit;

public class Demo2 {

    private  static int a =0;

    public static void main(String[] args) {

        new Thread(()->{
            while (a == 0){

            }
        }).start();

        try {
            TimeUnit.SECONDS.sleep(2);
        } catch (InterruptedException e) {
            e.printStackTrace();
        }

        a++;
        System.out.println(a);

    }
}

这时候代码是不会停止的,因为线程在死循环。

在这里插入图片描述

Volatile就可以去让这个线程停止,具有可见性。

package com.ys.VTest;

import java.util.concurrent.TimeUnit;

public class Demo2 {

    private volatile   static int a =0;

    public static void main(String[] args) {

        new Thread(()->{
            while (a == 0){

            }
        }).start();

        try {
            TimeUnit.SECONDS.sleep(2);
        } catch (InterruptedException e) {
            e.printStackTrace();
        }

        a++;
        System.out.println(a);

    }
}

指令重排

指令重排是指计算机并不会完全按照我们的代码顺序执行,处理器在进行指令重排的时候是考虑到数据之间的依赖性的,不会去乱排导致代码执行不下去。 Volatile可以去解决这个问题

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 1
    评论
评论 1
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值