浅聊JVM--基础版
  diRvFkCA5Zb8 2023年11月01日 177 0

浅聊JVM--基础版

一、来源


jvm共有三种

  • Sun公司: HotSpot使用最多
  • BEA:JRockit
  • IBM:J9VM

​ 今天我们主要了解的是Sun公司的HotSpot(关于HotSpot的爱恨情仇这里就不做过多解释了。)我们以前测试jdk是否安装成功,java的环境变量是否配置成功会使用java -version命令来检查。有一个细节大家可以看一下,cmd输入java -version回车后,可以查看jvm。上图

​ 大家可以看到,我们目前使用大都是Sun公司的产品。那么jvm处于什么位置呢?其实jvm是在操作系统之上,和硬件并无直接联系。

面试题:jvm、jdk、jre的区别?

二、概述


​ JVM是Java Virtual Machine的缩写,通俗来说也就是运行Java代码的容器。当项目启动时,会根据jvm相关配置参数,在计算机的内存中开启一片空间用于运行jvm,之后Java相关代码就会被加载进jvm中运行。那么为什么Java代码可以实现“一次编译,到处运行”的机制应该就与jvm有关了吧。

Java作为一种高级语言,要让计算机执行Java程序,也得需要经过编码-->编译-->运行步骤。但是Java编译程序并不能将Java源代码直接编译为计算机能识别的0/1指令,而是编译为字节码文件(.class)这时候就需要jvm将字节码文件翻译为与平台有关的0/1指令。所以有了jvm,Java程序就达到了“一次编译到处运行”的目的。所以其跨平台好的根本原因就是因为jvm的存在。但是JVM并不代表就可以执行class了,JVM执行.class还需要JRE下的lib类库的支持,尤其是rt.jar。

​ 它的内存空间包括方法区、堆、方法栈、本地方法栈、PC寄存器。(以上5块又成为运行时数据区)。我们刚才也提到,Java源代码经过Java编译程序。能够产生相应的.class文件,也就是字节码文件。字节码文件经过jvm中的解释器,再次编译成特定机器上的机器码指令。上图:

关于类加载器是如何工作的,大家可以参考文章:

[Java 类加载和类加载器 - 掘金 (juejin.cn)](https://juejin.cn/post/7226665757882236986)

面试题:什么是双亲委派原则?

面试题:JNI(Java Native Interface)Java本地方法接口中Native关键字的作用?

我们今天主要浅聊一下PC寄存器、方法区、栈、堆

三、PC寄存器


​ 程序计数器(Program Count Register):每个线程都有一个程序计数器,它是线程私有,是用来存储指向下一条指令的地址,也就是即将来执行的指令代码。在执行引擎读取下一条指令,是一个非常小的内存空间。(关于程序计数器更多的细节或者是在循环、判断等条件下PC是如何跳转的,大家可以参考《计算机组成原理》类资料了解更多底层原理。)

四、方法区


​ 方法区(Method Area):只是 JVM 规范中定义的一个概念,用于存储被 JVM 加载的类信息、常量、静态变量、即时编译器编译后的代码等数据。具体放在哪里,不同的实现可以放在不同的地方。HotSpot VM把 GC 分代收集扩展至方法区,即使用 Java 堆的永久代来实现方法区。(关于什么是永久代,请向下看)

面试题:一张白纸,画出对象实例化过程的内存图?

五、栈


​ 栈(Stack):里面放的是8大基本数据类型+对象引用+实例的方法。栈内存里,主管程序的运行,生命周期和线程同步,线程结束,栈内存也就释放了,栈内存结束,程序也就结束了。

​ 我们都知道在Java中,程序员不需要显示的去释放一个对象的内存的,而是由虚拟机自行回收垃圾,那么垃圾回收的过程不会涉及到栈(也就是说栈不存在垃圾回收的情况)

栈基于后进先出的特性,当执行完成后,会被弹出栈。

为什么main()是先执行,最后结束?(因为一开始是main()方法最先入栈,最后弹出,结束执行)

栈溢出的问题(StackOverflow)

看下一面一段程序:

jvm中的栈为:

六、堆


​ 堆:(Heap)运行时数据区,是被线程共享的一块内存区域,创建的对象和数组都保存在 Java 堆内存中,也是垃圾收集器进行 垃圾收集的最重要的内存区域。堆内存细分为3个区域:

  • 新生区(伊甸园去)--->Young/New(Eden Space)
  • 养老区(Old)
  • 永久区(Perm)

新生区又可以分为伊甸园区(Eden Space)、幸存0区、幸存1区,(幸存0区又可以称为SurvivorFrom区、幸存1区可以称为SurvivorTo区。From区和To区是可以来回动态交换的,具体什么原因,大家可以查阅资料)

伊甸园区:所有的对象都是在伊甸园区new出来的。

幸存者(0、1)区:在伊甸园区里经过一次轻GC后活下来的进入幸存(0、1)区,被清理掉的就死亡掉。(关于GC是如何进行垃圾回收的,大家可以自行查阅资料,这里只是简单了解)

养老区:在经过多次的轻GC处理后,幸存(0,1)区也都满了,会触发重GC来进行清理伊甸园区和幸存(0,1)区,那么进入养老区的就是重GC也未杀死的。若养老区也满了,就相当于内存已满,报OOM错误(java.lang.OutOfMemoryError:Java heap sapce)

听闻:99%的对象都是临时对象,能进入养老区的不多,所以OOM的错误也很少见。

面试题:工作中,遇到OOM了,你是怎么排查的?

1、尝试扩大堆内存空间,如果还满,可能有垃圾代码存在

2、分析内存,使用专业工具看哪里出了问题

[面试官:工作中遇到过 OOM 吗?你是怎么排查的? - 知乎 (zhihu.com)](https://zhuanlan.zhihu.com/p/165981061)

扩大内存空间参数:

(具体的参数设置大家自己查阅)

例如:

永久区:这个区域常驻内存,用来存放JDK自身携带的Class对象,Interface元数据,存储的是java运行时的一些环境或类信息,该区域不存在垃圾回收GC。关闭虚拟机就会释放这个内存。

关于永久区的更新换代:

jdk1.6之前:永久代,常量池在方法区

jdk1.7:永久代,但是慢慢退化了(去永久代)常量池在堆中

jdk1.8之后:无永久代,常量池在元空间

方法区又称非堆,本质还是堆,只是为了区分概念。

OK,结束!!!

资料参考:

狂神说Java

JVM入门笔记

【版权声明】本文内容来自摩杜云社区用户原创、第三方投稿、转载,内容版权归原作者所有。本网站的目的在于传递更多信息,不拥有版权,亦不承担相应法律责任。如果您发现本社区中有涉嫌抄袭的内容,欢迎发送邮件进行举报,并提供相关证据,一经查实,本社区将立刻删除涉嫌侵权内容,举报邮箱: cloudbbs@moduyun.com

  1. 分享:
最后一次编辑于 2023年11月08日 0

暂无评论

推荐阅读
  2Vtxr3XfwhHq   2024年05月17日   54   0   0 Java
  Tnh5bgG19sRf   2024年05月20日   110   0   0 Java
  8s1LUHPryisj   2024年05月17日   46   0   0 Java
  aRSRdgycpgWt   2024年05月17日   47   0   0 Java
diRvFkCA5Zb8