(.class字节码)类加载到内存之后,内存模型:(ps:.class文件可以通过javap 指令反编译成一个可读文件)

1.java栈,本地方法栈,程序计数器(每个线程私有)

看如下程序:

以该程序为例,运行该程序,jvm会分配给该程序一个线程,总体图示如下:

该线程在运行时候,java虚拟机会分配给该线程独立的java栈,而栈帧存在于栈中,存放的是 每一个方法运行时候需要的数据(每一个方法都有一个栈帧,栈帧存的是 局部变量表,操作数栈,动态链接,方法出口),上图有两个方法,即 jvm会分配两个栈帧。

首先入栈的是main方法的栈帧,当main方法new一个math对象时候,该栈帧当中存放了math的引用,而math对象是放在堆中,方法区中会放置Math类的.class文件,一些类的静态变量和常量也会放入方法区,比如:

然后main方法中调用了math方法(new了一个math对象),从而math方法的栈帧入栈,当math方法执行完毕之后,它的栈帧会弹出栈。

(使用javap指令反编译一下 Math.class)

查询jvm指令可知:iconst_1的含义是

(栈即使 操作数栈),istore_1的含义是

iconst_2,istore_2也一样。对应 int a=1 ;int b=2;

程序计数器也是每一个线程私有的,每个方法运行的时候,都有一个程序计数器,作用是告诉jvm接下来该运行哪一行代码,即是一个指针,如反编译后图的0,1,2,3......前四行代码都执行了,现在该运行4, 程序计数器放的内容是4

然后,执行int c=(a+b)*10,对应 iload_1,iload_2 ,含义如下,

对应结果(从局部变量表 装载到 操作数栈,PS:局部变量表中的值还在,只是复制到操作数栈中,下图显示不正确)

接着计算a+b,对应的是将b=2,a=1都弹出栈,进行+运算,然后将算出的结果3,放入操作数栈中,然后需要10,所以将10也压入栈

执行3*10的操作,需要将3和10均弹出栈进行乘的计算,计算的出30,再压回操作数栈中,然后将30弹出栈,进入局部变量表:

最后math方法执行方法执行完毕,会通过方法出口返回给main方法,并且,math方法的栈帧主动弹出栈销毁

本地方法栈是存放程序调用的native方法,(或者程序底层的native方法)

有个结论:java栈,本地方法栈,程序计数器是每一个线程私有的,而堆和方法区是所有线程所共享的。(堆和方法区共享是因为,其他线程也能创建相同的对象,比如math,也要用到方法区的一些内容)

附上两幅图:

2.堆和方法区的介绍(线程共享)

包括 新生代,老年代,元空间(MetaData space),(ps:方法区实际上不存在,只是一个逻辑上的概念,而元空间或者永久代<jdk 1.8之前>是方法区的具体代码实现)

老年代是新生代空间的两倍。

当程序创建对象的时候,对象会首先进入新生代的Eden中,如果新生代的Eden空间满了,Jvm会进行一些小GC,将一些没有引用指向的对象,即垃圾对象清除掉。其他的不能GC的对象(即还有引用指向的对象)会被移到Suvivor区域,如果第一个Suvivor区(From)也满了,也会进行GC,GC之后,会将第一个Suvivor剩下的所有对象复制到第二个Suvivor区域(To)。

程序如果继续创建对象,会继续对Eden GC,这时候,其他不能GC的对象会被移到Survivor第二个区域中(To区域),第二个区域和第一个区域角色互换,重复上面操作,然后再一轮角色又互换。这样经过好几轮.........................................

如果JVM这样好几次操作(JDK1.8默认15次,也可设置)之后,还有些对象没有被GC,这些对象会被放入老年代Old Generation。如果程序再创建对象,新生代和老年代都满了,JVM会对老年代进行一次full GC,清楚无效的对象,Full GC可能会使JVM暂停(对用户而言程序失去响应,但是内部还是再做GC),而JVM调优带到的效果是:无非是使Full gc少执行,而小GC也少执行。

转载于:https://www.cnblogs.com/chengwu1996/p/10466977.html

JVM内存模型分析(一个程序运行的例子)相关推荐

  1. java实现初始化三角形数组_Java 数组、多维数组,动态、静态初始化,数组JVM内存模型分析...

    Java 数组.多维数组,动态.静态初始化,数组JVM内存模型分析 什么是数组 所谓数组,是具有相同数据类型的若干变量或者数据按照一定排序规则组合起来的一种数据存储格式.数组中的数据称为数组元素,我们 ...

  2. 面试回答,JVM内存模型/内存空间:运行时数据区

    发布于个人公众号,打开微信,搜索MelodyJerry即可 本文由作者原文 [JVM|内存模型] Java虚拟机的内存模型?也就这7个而已 修改而来,可点击左下角阅读原文. JVM内存模型/内存空间 ...

  3. Java JVM内存模型

    简述JVM内存模型 线程私有的运行时数据区: 程序计数器.Java 虚拟机栈.本地方法栈. 线程共享的运行时数据区:Java 堆.方法区. 简述程序计数器 程序计数器表示当前线程所执行的字节码的行号指 ...

  4. JVM内存模型及常见问题

    JVM内存模型及常见问题 JVM内存模型 堆内存模型展示 对象创建过程 常见问题及解答 如何理解各种GC 为什么需要survivor区?只有Eden不行吗? 为什么需要两个S区? 为什么Eden:s1 ...

  5. java内存图怎么画,jvm内存模型怎么画?简单美观的模型图制作软件

    jvm内存模型是什么呢?jvm内存模型是一类用于描述由java栈.java堆.方法区.本地方法栈和程序计数器各部件构成的Java程序.在执行过程中,由jvm管理的不同数据区域的一类图表.jvm内存模型 ...

  6. JVM内存模型和类加载运行机制

    JVM内存模型和类加载运行机制 JVM内存模型 运行一个 Java 应用程序,必须要先安装 JDK 或者 JRE 包.因为 Java 应用在编译后会变成字节码,通过字节码运行在 JVM 中,而 JVM ...

  7. JVM虚拟机学习 - JVM类加载,JVM内存模型,JVM性能分析工具

    JVM虚拟机 二 JVM类加载 类的生命周期 加载: ​ 加载class文件到二进制字节流,然后再将二进制字节流转化为方法区的运行时数据结构,生成一个对应的Class对象作为类各种数据的访问入口. 链 ...

  8. 膜拜大佬!JVM性能调优——JVM内存模型和类加载运行机制

    一.JVM内存模型 运行一个 Java 应用程序,必须要先安装 JDK 或者 JRE 包.因为 Java 应用在编译后会变成字节码,通过字节码运行在 JVM 中,而 JVM 是 JRE 的核心组成部分 ...

  9. 详细分析JVM内存模型

    JVM内存模型 JAVA的主要特点是其著名的WOTA(write once, run anywhere):"编写一次,随处运行".为了应用它,Sun Microsystems创建了 ...

最新文章

  1. 高防服务器如何进行防御措施
  2. 卸载重装svn后原来项目不受管理,版本不对应还是,升级工作副本解决?
  3. Visual Studio现可使用EditorConfig
  4. [GAN学习系列] 初识GAN
  5. eureka需要替换吗_nacos无缝替换eureka
  6. linux打包启动这着的文件,linux – 打开一个RPM文件并重新打包它
  7. 多媒体计算机技术英文,掌握多媒体计算机技术中各专业语句的英文表达方式.doc...
  8. 【ML小结7】贝叶斯分类器:朴素、半朴素贝叶斯
  9. linux libaio介绍
  10. 解决Ubuntu无法定位软件包问题
  11. C++求出200以内的所有质数(素数),并按每行5个输出在屏幕上。
  12. 计算机安全模式无法启动,我的电脑安全模式可以进去,为什么不能正常起动呢?...
  13. 原神3.2剧情服搭建教程
  14. SQL 四大功能DDL/DML/DCL/TCL
  15. “数据”到底是资产还是负债?
  16. 写给大数据初学者的话——转自lxw的大数据田地
  17. 两种方法判断是否为移动端访问,跳转到对应wap页面
  18. Ant Design Vue Table 组件合并单元格
  19. OMS系统是什么?主要功能和优势有哪些?
  20. UbuntuX86_64交叉编译opencv为arm64架构

热门文章

  1. linux程序改ip地址吗,如何在Linux中从C设置IP地址
  2. Spring Security UserDetail
  3. python *args **kargs
  4. javascript Nested functions
  5. C++ 对比 C语言
  6. linux iptables
  7. 数据算法之二叉树查找(BinaryTreeL Search)的Java实现
  8. 计算机类专业权威解读,09计算机考研统考大纲权威解读之操作系统
  9. linux 64位module内联汇编,@yuanbor: Linux内联汇编总结
  10. 什么是对象存储OSS-对象存储 OSS > 产品简介 > 什么是对象存储OSS