JVM详解(5)--JMM

1.JMM定义
JMM(Java内存模型Java Memory Model,简称JMM)本身是一种抽象的概念并不真实存在它描述的是一组规范或规则,通过这组规范定义了程序中各个变量(包括实例字段,静态字段和构成数组对象的元素)的访问方式。
2.JMM关于同步的规定:
(1)线程解锁前,必须把共享变量的值刷新回主内存。
(2)线程加锁前,必须读取主内存的最新值到自己的工作内存。
(3)j加锁解锁是同一把锁。
3.线程对共享变量的操作过程
由于JVM运行程序的实体是线程,而每个线程创建时JVM都会为其创建一个工作内存(有些地方称为栈空间),工作内存时每个线程的私有数据区域,而Java内存模型规定所有变量存储在主内存,主内存是共享内存区域,所有线程都可以访问,但线程对变量的操作(读取赋值等)必须在工作内存中进行,首先要将变量从主内存拷贝到线程自己的工作内存,然后对变量进行操作,操作完成后在将变量写回主内存,不能直接操作主内存中的变量,各个线程中的工作内存中存储着内存中的变量副本拷贝,因此不同的线程间无法访问对方的工作内存,线程间的通信(传值)必须通过主内存来完成,其简要访问过程如下图:
JVM详解(5)--JMM_第1张图片
4.代码实现

public class visibility {
    public static void main(String[] args) {
        Resources resources = new Resources();
        new Thread(()->{
            System.out.println(Thread.currentThread().getName()+" is coming!");
            try {
                Thread.sleep(3000);
            } catch (InterruptedException e) {
                e.printStackTrace();
            }
            resources.change();
            System.out.println(Thread.currentThread().getName()+" is updating num,num="+resources.num);
        }).start();
        while (resources.num == 10){

        }
        System.out.println(Thread.currentThread().getName()+"is over");
    }
}
class Resources{
    public int num = 10;
    public void change(){
        this.num = 100;
    }
}

JVM详解(5)--JMM_第2张图片
为什么上面的程序没有停止,明明Thread-0修改了num的值,而且还打印了出来,为什么main线程读到的还是10。这是因为涉及到JMM的内存可见性问题,虽然Thread-0在自己的工作内存中修改了num,并把它写入到了主内存,但是Thread-0线程的修改对main线程是不可见的,所以main线程的工作内存中num值还是10。

5.怎么修改呢?
非常简单,共享变量只需用volatile关键字修饰就行。volatile关键字的作用就是每次main线程读取num的时候,都会先把主内存中的num更新到自己的工作内存,那么每次读的num值都是最新的,从而实现了内存可见性。

你可能感兴趣的:(JVM)