zoukankan      html  css  js  c++  java
  • JVM的内存模型

     参考博客:https://www.cnblogs.com/xdecode/p/8948277.html

    内存模型:每一个线程有一个工作内存和主存独立,工作内存存放主存中变量的值的拷贝


    当数据从主内存复制到工作存储时,必须出现两个动作:第一,由主内存执行的读(read)操作;第二,由工作内存执行的相应的load操作;当数据从工作内存拷贝到主内存时,也出现两个操作:第一个,由工作内存执行的存储(store)操作;

    第二,由主内存执行的相应的写(write)操作每一个操作都是原子的,即执行期间不会被中断。对于普通变量,一个线程中更新的值,不能马上反应在其他变量中如果需要在其他线程中立即可见,需要使用 volatile 关键字

    可见性:一个线程修改了变量,其他线程可以立即知道,在本线程内,操作都是有序的在线程外观察,操作都是无序的。(指令重排 或 主内存同步延时)
    保证可见性的方法:volatile,synchronized (unlock之前,写变量值回主存),final(一旦初始化完成,其他线程就可见),LOCK锁等


    指令重排:指令重排是指在程序执行过程中, 为了性能考虑, 编译器和CPU可能会对指令重新排序.。保证线程内串行语义,破坏线程间的有序性,编译器不考虑多线程间的语义

    写后读 a = 1;b = a; 写一个变量之后,再读这个位置。
    写后写 a = 1;a = 2; 写一个变量之后,再写这个变量。
    读后写 a = b;b = 1; 读一个变量之后,再写这个变量。
    以上语句不可重排

    可重排: a=1;b=2;

    指令重排遵循的基本原则(Happen-Before先行发生规则):
    程序顺序原则:一个线程内保证语义的串行性
    volatile规则:volatile变量的写,先发生于读
    锁规则:解锁(unlock)必然发生在随后的加锁(lock)前
    传递性:A先于B,B先于C 那么A必然先于C
    线程的start方法先于它的每一个动作
    线程的所有操作先于线程的终结(Thread.join())
    线程的中断(interrupt())先于被中断线程的代码
    对象的构造函数执行结束先于finalize()方法

  • 相关阅读:
    kotlin的三目运算
    oracle存储过程相关整理
    LInux系统部署Java项目
    Idea集成vue
    java执行查询存储过程
    访问分享文件路径方法
    java计算程序运行时间
    IntelliJ IDEA 最新版 2019.1 安装与激活
    oracel中将子表中一个字段多条数据合并到主表的sql
    oracel根据条件不同,统计数据sql
  • 原文地址:https://www.cnblogs.com/cheng21553516/p/11220296.html
Copyright © 2011-2022 走看看