JVM学习-内存结构(一)

一、引言

学前了解:

1.什么是JVM

1.1定义

Java Virtual Machine ,Java 程序的运行环境(Java 二进制字节码的运行环境)。

  • 好处

    • 一次编译,处处执行

    • 自动的内存管理,垃圾回收机制

    • 数组下标越界检查

  • 比较 JVM、JRE、JDK 的关系如下图所示

2.学JVM有什么用

  • 理解底层实现原理,比如:自动装箱、自动拆箱是怎么实现的,反射是怎么实现的,垃圾回收机制是怎么回事等待,JVM 是必须掌握的。

  • 中高级程序员必备技能,同时也是面试必备

3.常见的JVM

  • 我们学习的是HotSpot 版本的虚拟机。

4.学习路线

  • ClassLoader:Java 代码编译成二进制后,会经过类加载器,这样才能加载到 JVM 中运行。 Method Area:类是放在方法区中。 Heap:类的实例对象。 当类调用方法时,会用到 JVM Stack、PC Register、本地方法栈。 方法执行时的每行代码是有执行引擎中的解释器逐行执行,方法中的热点代码频繁调用的方法,由 JIT 编译器优化后执行,GC 会对堆中不用的对象进行回收。需要和操作系统打交道就需要使用到本地方法接口。

二、内存结构

1.内存结构的组成有五部分

2.程序计数器

2.1定义与特点

2.2解释:

程序计数器CPU中的寄存器实现

3.java虚拟机栈

3.1定义

1)栈与栈帧,栈就是每个线程所需要的空间(总的),栈帧是每个方法运行时所需要的

2)用程序解释如下:

3.2问题辨析

1)垃圾回收是否涉及栈内存?

不会。栈内存在每次方法调用结束后都会自动释放,不需要垃圾回收处理。

2)栈内存分配越大越好吗?

不是。因为物理内存是一定的,栈内存越大,可以支持更多的递归调用,但是可执行的线程数就会越少。运行时可以指定栈内存大小,有默认值。

3)方法呢的局部变量是否线程安全
  • 如果方法内部的变量没有逃离方法的作用访问,它是线程安全的

  • 如果是局部变量引用了对象,并逃离了方法的访问,那就要考虑线程安全问题。

  • static修饰的变量会产生线程安全因为是公共的,其他的不会因为是每个线程私有的。

  • 线程不安全例子如下(图中m2,m3都是不安全的):

    4)栈溢出错误:
    java.long.StackOverflowError
    5)运行时设置栈内存大小

3.3线程诊断(案例)

1)案例一:CPU占用过多

要求排查出是哪行代码导致的CPU占用过多(一个java代码运行在linux中)

nohup 让java代码在后台运行

top 命令可以检测到后台进程对CPU的使用对内存的占用情况

得到有问题的进程号是32655,但不能知道是进程中的那个线程

用PS命令定位线程,H是打印线程数 -eo是规定输出的信息

使用jdk提供的jstack命令查看具体进程中的,jstack +进程ID

可以列出所有的线程列出,再根据ps命令得到的线程号转为16进制即可看到具体有问题的行数(ps得到的为10进制),32665转16进制为7F99

总结

2)案例二:程序运行长时间没有结果

即线程死锁,还是使用jstack命令

可以看到,在最后有说明是用户线程1和用户线程0产生死锁分别在29行和21行

相关推荐
极客先躯6 分钟前
说说高级java每日一道面试题-2025年2月13日-数据库篇-请说说 MySQL 数据库的锁 ?
java·数据库·mysql·数据库的锁·模式分·粒度分·属性分
程序员侠客行9 分钟前
Spring事务原理 二
java·后端·spring
小猫猫猫◍˃ᵕ˂◍23 分钟前
备忘录模式:快速恢复原始数据
android·java·备忘录模式
liuyuzhongcc32 分钟前
List 接口中的 sort 和 forEach 方法
java·数据结构·python·list
五月茶36 分钟前
Spring MVC
java·spring·mvc
sjsjsbbsbsn1 小时前
Spring Boot定时任务原理
java·spring boot·后端
yqcoder1 小时前
Express + MongoDB 实现在筛选时间段中用户名的模糊查询
java·前端·javascript
菜鸟蹦迪1 小时前
八股文实战之JUC:ArrayList不安全性
java
2501_903238651 小时前
Spring MVC配置与自定义的深度解析
java·spring·mvc·个人开发
逻各斯1 小时前
redis中的Lua脚本,redis的事务机制
java·redis·lua