最新java内存模型_java内存模型

最新java内存模型_java内存模型1.Java内存模型Java内存模型是每个java程序员必须掌握理解的,这是Java的核心基础,对我们编写代码特别是并发编程时有很大帮助。由于Java程序是交由JVM执行的,所以我们在谈Java内存区域划分的时候事实上是指JVM内存区域划分。1.1.Java程序执行流程回顾如图所示首先Java源代码文件(.java后缀)会被Java编译器编译为字节码文件(.class后缀),然后由JVM中…

大家好,又见面了,我是你们的朋友全栈君。

1.   Java内存模型

Java内存模型是每个java程序员必须掌握理解的,这是Java的核心基础,对我们编写代码特别是并发编程时有很大帮助。由于Java程序是交由JVM执行的,所以我们在谈Java内存区域划分的时候事实上是指JVM内存区域划分。

1.1. Java程序执行流程回顾

cb4e3fc3f817449393dd0724b8d1abb3.png

如图所示

首先Java源代码文件(.java后缀)会被Java编译器编译为字节码文件(.class后缀),

然后由JVM中的类加载器加载各个类的字节码文件,

加载完毕之后,交由JVM执行引擎执行。

Java内存模型指的就是Runtime Data Area(运行时数据区),即程序执行期间用到的数据和相关信息保存区。

1.2. Java内存模型

根据 JVM 规范,JVM 内存共分为虚拟机栈、堆、方法区、程序计数器、本地方法栈五个部分。结构如下图:

bcd64677c931dd68e42e3394fd0a24d9.png

1.2.1.   PC程序计数器:

l  每个线程对应有一个程序计数器。

l  各线程的程序计数器是线程私有的,互不影响,是线程安全的。

l  程序计数器记录线程正在执行的内存地址,以便被中断线程恢复执行时再次按照中断时的指令地址继续执行

1.2.2.   Java栈JavaStack(虚拟机栈JVM Stack):

l  每个线程会对应一个Java栈;

l  每个Java栈由若干栈帧组成;

l  每个方法对应一个栈帧;

l  栈帧在方法运行时,创建并入栈;方法执行完,该栈帧弹出栈帧中的元素作为该方法返回值,该栈帧被清除;

l  栈顶的栈帧叫活动栈,表示当前执行的方法,才可以被CPU执行;

l  线程请求的栈深度大于虚拟机所允许的深度,将抛出StackOverflowError异常;

l  栈扩展时无法申请到足够的内存,就会抛出OutOfMemoryError异常;

a7aedabb21a3c125dc0fe1a490f49deb.png

1.2.3.   方法区MethodArea

l  方法区是Java堆的永久区(PermanetGeneration)

l  方法区存放了要加载的类的信息(名称、修饰符等)、类中的静态常量、类中定义为final类型的常量、类中的Field信息、类中的方法信息,

l  方法区是被Java线程共享的

l  方法区要使用的内存超过其允许的大小时,会抛出OutOfMemoryError: PremGen space的错误信息。

1.2.4.   常量池ConstantPool:

l  常量池是方法区的一部分。

l  常量池中存储两类数据:字面量和引用量。

字面量:字符串、final变量等。

引用量:类/接口、方法和字段的名称和描述符,

l  常量池在编译期间就被确定,并保存在已编译的.class文件中

1.2.5.   本地方法栈Native Method Stack:

l  本地方法栈和Java栈所发挥的作用非常相似,区别不过是Java栈为JVM执行Java方法服务,而本地方法栈为JVM执行Native方法服务。

l  本地方法栈也会抛出StackOverflowError和OutOfMemoryError异常。

1.3. Java内存模型工作示意图

3d600a0cb4e3843783aa06651fea3277.png

1)         首先类加载器将Java代码加载到方法区

2)         然后执行引擎从方法区找到main方法

3)         为方法创建栈帧放入方法栈,同时创建该栈帧的程序计数器

4)         执行引擎请求CPU执行该方法

5)         CPU将方法栈数据加载到工作内存(寄存器和高速缓存),执行该方法

6)         CPU执行完之后将执行结果从工作内存同步到主内存

线程计算的时候,原始的数据来自内存,在计算过程中,有些数据可能被频繁读取,这些数据被存储在寄存器和高速缓存中,当线程计算完后,这些缓存的数据在适当的时候应该写回内存。

当个多个线程同时读写某个内存数据时,就会产生多线程并发问题,要解决这些问题就涉及到多线程编程三个特性:原子性,有序性,可见性。

版权声明:本文内容由互联网用户自发贡献,该文观点仅代表作者本人。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如发现本站有涉嫌侵权/违法违规的内容, 请联系我们举报,一经查实,本站将立刻删除。

发布者:全栈程序员-站长,转载请注明出处:https://javaforall.net/156619.html原文链接:https://javaforall.net

(0)
全栈程序员-站长的头像全栈程序员-站长


相关推荐

  • 如何完成域名和ip地址的绑定

    如何完成域名和ip地址的绑定

    2021年9月20日
    60
  • python 命名规范

    python 命名规范python变量命名规则:变量名只能包含字母、数字和下划线。变量名可以字母或下划线开头,但不能以数字开头,例如,可将变量命名为message_1,但不能将其命名为1_message。 变量名不能包含空格,但可使用下划线来分隔其中的单词。例如,变量名greeting_message可行,但变量名greetingmessage会引发错误。 不要将Python关键字和函数名用作变量名,即不要…

    2022年6月25日
    35
  • mysql省市区递归查询_mysql 递归查询

    mysql省市区递归查询_mysql 递归查询1、创建表:DROPTABLEIFEXISTS`t_areainfo`;CREATETABLE`t_areainfo`(`id`int(11)NOT”AUTO_INCREMENT,`level`int(11)DEFAULT”,`name`varchar(255)DEFAULT”,`parentId`int(11)DEFAULT”,`status`i…

    2022年6月15日
    36
  • C# Lock 解读

    C# Lock 解读

    2021年8月15日
    57
  • jsonp跨域原理简单总结_jsonp的工作原理

    jsonp跨域原理简单总结_jsonp的工作原理JavaScript是一种在Web开发中经常使用的前端动态脚本技术。在JavaScript中,有一个很重要的安全性限制,被称为“Same-OriginPolicy”(同源策略)。这一策略对于JavaScript代码能够访问的页面内容做了很重要的限制,即JavaScript只能访问与包含它的文档在同一域下的内容。JavaScript这个安全策略在进行多iframe或多窗口编程、以及Ajax

    2025年6月10日
    3
  • extremedb同步mysql_数据库选型之内存数据库eXtremeDB[通俗易懂]

    extremedb同步mysql_数据库选型之内存数据库eXtremeDB[通俗易懂]刘勇Email:lyssym@sina.com简介鉴于内存数据库访问速率快的特点,本文分别从单线程、多线程(并发访问)和多线程读/写混合访问角度对eXtremeDB数据库读写速率展开测试。需要指出的是,本文读取操作包含将数据读取后,并在控制台显示出来。测试结果表明:eXtremeDB在单一读/写访问时,速率大约在10w条/s,其速率是比较快的;同时相对单线程来说,多线程读或者写操作并发访问…

    2022年10月14日
    3

发表回复

您的邮箱地址不会被公开。 必填项已用 * 标注

关注全栈程序员社区公众号