JVM(二)JVM内存模型

简介: JVM(二)JVM内存模型

Java虚拟机的数据区域

JVM.jpg

JVM内存参数设置

  • -Xss:每个线程的栈大小,默认1m
  • -Xms:初始堆大小,默认物理内存的1/64
  • -Xmx:最大堆大小,默认物理内存的1/4
  • -Xmn:新生代大小
  • -XX:NewSize:设置新生代初始大小
  • -XX:NewRatio:默认2表示新生代占年老代的1/2,占整个堆内存的1/3。
  • -XX:SurvivorRatio:默认8表示一个survivor区占用1/8的Eden内存,即1/10的新生代内存。
  • -XX:MaxMetaspaceSize: 设置元空间最大值, 默认是-1, 即不限制, 或者说只受限于本地内存大小。
  • -XX:MetaspaceSize: 指定元空间触发full gc的初始阈值(元空间无固定初始大小), 以字节为单位,默认是21M,达到该值就会触发full gc进行类型卸载, 同时收集器会对该值进行调整: 如果释放了大量的空间, 就适当降低该值; 如果释放了很少的空间, 那么在不超过-XX:MaxMetaspaceSize(如果设置了的话) 的情况下, 适当提高该值。这个跟早期jdk版本的-XX:PermSize参数意思不一样,-XX:PermSize代表永久代的初始容量。

由于调整元空间的大小需要Full GC,这是非常昂贵的操作,如果应用在启动的时候发生大量Full GC,通常都是由于永久代或元空间发生了大小调整,基于这种情况,一般建议在JVM参数中将MetaspaceSizeMaxMetaspaceSize设置成一样的值,并设置得比初始值要大,对于8G物理内存的机器来说,一般可以将这两个值都设置为256M

1. 程序计数器

程序计数器是一块较小的内存空间,它可以看做是当前线程所执行的字节码的行号指示器。

由于Java虚拟机的多线程是通过线程轮流切换、分配处理器执行时间来实现的,如果一个线程在执行一段代码,在执行中间被切换到了其他线程,进入了等待,那当该线程切换回来时,如果确定当前线程执行的位置呢,这时候就是程序计数器发挥作用了,他存储的值可以使得字节码解释器可以得知下一条需要执行的字节码指令。从这里我们也可以知道,每个线程都应该有一个独立的程序计数器,不然要是多个线程共享一个程序计数器的话,那我怎么知道这个要执行的字节码指令是哪个线程的勒。

此内存区域是唯一一个在《Java虚拟机规范》中没有规定任务OutOfMemoryError情况的区域

2. Java虚拟机栈

和程序计数器一样,Java虚拟机栈也是线程私有的,它的生命周期与线程相同。

虚拟机栈描述的是Java方法执行线程内存模型:每个方法被执行的时候,Java虚拟机都会同步创建一个栈帧。

  • 栈帧:Java虚拟机栈所分配的一块内存区域,它里面包含
  • 局部变量表 存放各种基础数据类型,对象引用,returnAddress
  • 操作数栈
  • 动态连接
  • 方法出口

每一个方法调用完毕,就对应着一个栈帧在虚拟机栈从入栈到出栈的过程

异常:StackOverflowErrorOutOfMemoryEroor

3. 本地方法栈

与虚拟机栈相似,一个执行Java方法,一个执行本地方法

4. 堆

  • 共享情况:所有线程共享
  • 创建时间:虚拟机启动时创建
  • 存储内容:存放了几乎所有的对象实例
  • 扩展性:可扩展,通过参数-Xmx-Xms设定
  • 异常:OutOfMemoryError

各个区域的分配,回收的细节在下几篇笔记中记录

5.方法区

  • 共享情况:所有线程共享
  • 创建时间:虚拟机启动时创建
  • 存储内容:存储已被虚拟机加载的类信息、常量、静态变量、即使编译器编译后的代码缓存
  • 元空间:主要存储类型信息
  • 运行时常量池:存储类的常量池表
  • 参数:-XX:MetaspaceSize
  • 异常:OutOfMemoryError

6.直接内存

直接内存并非虚拟机运行时数据区的一部分,它是NIO中,使用Native函数库直接分配的堆外内存,这块内存使用DirectByteBuffer进行操作

此内存虽不受到Java堆大小的限制,但仍在本机内存中,所以在配置-Xmx等参数时不要忽略掉了该内存从而导致OutOfMemoryError

代码案例

安装Visual GC查看运行时内存状态

  1. 找到对应jdk版本的插件url 如jdk7:https://visualvm.github.io/archive/uc/8u40/updates.xml.gz
  2. 在命令行输入命令jvisualvm打开Java VisualVM
  3. 选择工具->插件->设置
  4. 编辑Java Visual VM 插件中心配置,修改url为刚才找到的url:https://visualvm.github.io/archive/uc/8u40/updates.xml.gz
  5. 在可用插件中选择Visual GC 安装
  6. 安装完成,重启,选择java进程时便可看到Visual GC页

测试栈溢出

/**
 * 测试栈溢出 -Xss128k 默认为1m
 * @author Zijian Liao
 */
public class StackOverflowTest {
    static int count = 0;
    static void redo() {
        count++;
        redo();
    }
    public static void main(String[] args) {
        try {
            redo();
        } catch (Throwable t) {
            t.printStackTrace();
            System.out.println(count);
        }
    }
}

测试内存参数

/**
 * 测试内存参数 -Xms64m -Xmx64m -Xmn30m -XX:NewRatio=1 -XX:MaxMetaspaceSize=256m -XX:MetaspaceSize=256m
 *
 * @author Zijian Liao
 * @since 1.0
 */
public class MemoryTest {
    static class Memory{
        byte[] bytes = new byte[1024*1024];//1m
    }
    public static void main(String[] args) throws InterruptedException {
        List<Memory> memoryList = new ArrayList<>();
        for(;;){
            memoryList.add(new Memory());
            Thread.sleep(1000);
        }
    }
}
目录
相关文章
|
7月前
|
Arthas 存储 算法
深入理解JVM,包含字节码文件,内存结构,垃圾回收,类的声明周期,类加载器
JVM全称是Java Virtual Machine-Java虚拟机JVM作用:本质上是一个运行在计算机上的程序,职责是运行Java字节码文件,编译为机器码交由计算机运行类的生命周期概述:类的生命周期描述了一个类加载,使用,卸载的整个过类的生命周期阶段:类的声明周期主要分为五个阶段:加载->连接->初始化->使用->卸载,其中连接中分为三个小阶段验证->准备->解析类加载器的定义:JVM提供类加载器给Java程序去获取类和接口字节码数据类加载器的作用:类加载器接受字节码文件。
661 55
|
2月前
|
存储 缓存 Java
我们来说一说 JVM 的内存模型
我是小假 期待与你的下一次相遇 ~
218 4
|
2月前
|
存储 缓存 算法
深入理解JVM《JVM内存区域详解 - 世界的基石》
Java代码从编译到执行需经javac编译为.class字节码,再由JVM加载运行。JVM内存分为线程私有(程序计数器、虚拟机栈、本地方法栈)和线程共享(堆、方法区)区域,其中堆是GC主战场,方法区在JDK 8+演变为使用本地内存的元空间,直接内存则用于提升NIO性能,但可能引发OOM。
|
8月前
|
Arthas 监控 Java
Arthas memory(查看 JVM 内存信息)
Arthas memory(查看 JVM 内存信息)
646 6
|
缓存 Prometheus 监控
Elasticsearch集群JVM调优设置合适的堆内存大小
Elasticsearch集群JVM调优设置合适的堆内存大小
2037 1
|
11月前
|
存储 设计模式 监控
快速定位并优化CPU 与 JVM 内存性能瓶颈
本文介绍了 Java 应用常见的 CPU & JVM 内存热点原因及优化思路。
1044 166
|
9月前
|
存储 缓存 算法
JVM简介—1.Java内存区域
本文详细介绍了Java虚拟机运行时数据区的各个方面,包括其定义、类型(如程序计数器、Java虚拟机栈、本地方法栈、Java堆、方法区和直接内存)及其作用。文中还探讨了各版本内存区域的变化、直接内存的使用、从线程角度分析Java内存区域、堆与栈的区别、对象创建步骤、对象内存布局及访问定位,并通过实例说明了常见内存溢出问题的原因和表现形式。这些内容帮助开发者深入理解Java内存管理机制,优化应用程序性能并解决潜在的内存问题。
399 29
JVM简介—1.Java内存区域
|
9月前
|
缓存 监控 算法
JVM简介—2.垃圾回收器和内存分配策略
本文介绍了Java垃圾回收机制的多个方面,包括垃圾回收概述、对象存活判断、引用类型介绍、垃圾收集算法、垃圾收集器设计、具体垃圾回收器详情、Stop The World现象、内存分配与回收策略、新生代配置演示、内存泄漏和溢出问题以及JDK提供的相关工具。
JVM简介—2.垃圾回收器和内存分配策略
|
9月前
|
存储 设计模式 监控
如何快速定位并优化CPU 与 JVM 内存性能瓶颈?
如何快速定位并优化CPU 与 JVM 内存性能瓶颈?
252 0
如何快速定位并优化CPU 与 JVM 内存性能瓶颈?
|
10月前
|
存储 算法 Java
JVM: 内存、类与垃圾
分代收集算法将内存分为新生代和老年代,分别使用不同的垃圾回收算法。新生代对象使用复制算法,老年代对象使用标记-清除或标记-整理算法。
145 6