详细了解JVM运行时内存( 四 )

  • -XX:MinMetaspaceFreeRatio,在GC之后,最小的Metaspace剩余空间容量的百分比,减少为分配空间所导致的垃圾收集 。
  • -XX:MaxMetaspaceFreeRatio,在GC之后,最大的Metaspace剩余空间容量的百分比,减少为释放空间所导致的垃圾收集 。
  • 6.方法区6.1方法区的理解概念:
    元空间、永久代是方法区具体的落地实现 。方法区看作是一块独立于Java堆的内存空间,它主要是用来存储所加载的类信息的,方法区是线程共享的 。
    特点:
    • 方法区与堆一样是各个线程共享的内存区域 。
    • 方法区在JVM启动的时候就会被创建,并且它实际的物理内存空间和Java堆一样可以是不连续的 。
    • 方法区的大小跟堆一样,可以选择固定的大小或者动态变化 。
    • 方法区的大小决定了系统可以保存多少个类,如果系统定义了太多的类,导致方法区溢出,虚拟机仍然会报OOM异常 。
    • 关闭虚拟机就会释放方法区区域 。
    6.2 方法区结构类加载器将Class文件加载到内存以后,将类的信息存储到方法区中 。
    方法区中存储的内容:
    • 类型信息(域信息、方法信息)
    • 运行时常量池

    详细了解JVM运行时内存

    文章插图
    类型信息
    • 这个类型的完整有效名称(全名 = 包名.类名)
    • 这个类型直接父类的完整有效名(对于 interface或是java.lang. Object,都没有父类)
    • 这个类型的修饰符( public, abstract,final的某个子集)
    • 这个类型直接接口的一个有序列表
    域信息
    域信息,即为类的属性,成员变量JVM必须在方法区中保存类所有的成员变量相关信息及声明顺序 。域的相关信息包括:域名称、域类型、域修饰符(pυblic、private、protected、static、final、volatile、transient的某个子集)
    方法信息
    1. 方法名称方法的返回类型(或void)
    2. 方法参数的数量和类型(按顺序)
    3. 方法的修饰符public、private、protected、static、final、synchronized、native,、abstract的一个子集
    4. 方法的字节码bytecodes、操作数栈、局部变量表及大小( abstract和native方法除外)
    5. 异常表( abstract和 native方法除外) 。每个异常处理的开始位置、结束位置、代码处理在程序计数器中的偏移地址、被捕获的异常类的常量池索引
    6.3 方法区设置方法区的大小不必是固定的,可以根据应用的需要动态调整
    • jdk7及之前
      • 通过-xx:Permsize来设置永久代初始分配空间 。
      • -XX:MaxPermsize来设定永久代最大可分配空间 。64位的机器默认是82M 。当JVM加载的类信息容量超过了这个值,会报OOM异常:PermGen space 。
    • jdk8及以后
      • 元数据区大小可以使用参数 -XX:MetaspaceSize 和 -XX:MaxMetaspaceSize指定 。但是元数据区的 -XX:MaxMetaspaceSize默认是-1即没有限制,不设置可以使用系统剩余所有内存 。
      • 如果元数据区发生溢出,虚拟机一样会抛出异常OutOfMemoryError:Metaspace
    7.运行时常量池字节码文件中,内部包含了常量池 。
    方法区中,内部包含了运行时常量池 。
    常量池:存放了编译期间产生的各种字面量和符号引用 。
    运行时常量池:是常量池表在运行时的一种表现形式 。
    编译后的字节码文件中包含了类型信息、域信息、方法信息等 。通过ClassLoader将字节码文件的常量池中的信息加载到内存中,存储在了方法区的运行时常量池中 。
    详细了解JVM运行时内存

    文章插图
    常量池,可以看做是一张表,虚拟机指令根据这张常量表找到要执行的类名、方法名、参数类型、字面量等类型 。

    经验总结扩展阅读