zoukankan      html  css  js  c++  java
  • 第十六篇 JVM之运行时数据区<12>: 方法区

    一、方法区(Method Area)

      JVM虚拟机规范中虽然在逻辑上将方法区描述为堆区的一部分,但对于HotSpot虚拟机而言,还有一个别名Non-Heap(非堆),目的是与堆区分开。方法也是线程共享的区域,在JVM启动的时候被创建,并且,和堆区一样可以是逻辑上连续,物理上不连续的区域。
    方法区包含了四部分内容;

    • 类型信息:类的版本、接口、字段、方法等描述信息。
    • 运行时常量池:Class文件加载后的常量池数据(各种字面量和符号引用),直接引用,字符串常量池(JDK8移入Java堆中)。字面量:各种文本字符串、final常量值、基本类型数据数值等,如描述Object类的"()Ljava/lang/Object"字符串就是一个字面量。符号引用:如cp info #44 <toString>,#44就是对toString方法的符号引用。
    • JIT代码缓存:即时编译器编译后的代码缓存。
    • 静态变量:类中的静态变量,JDK7以后移入Java堆中,并且将静态变量存在类型数据对应的java.lang.Class实例里面,下面是引自《深入理解Java虚拟机》的图。


    二、方法区实现的变化

      JDK8以前,很多人把方法区称为永久代(Permanent Generation),实际上两者并不等价,这一说法是因为HotSpot虚拟机将垃圾收集分代设计将区域扩展到方法区,或者说永久代是方法区的实现。由于永久代更容易OOM,所以该设计在JDK7以后逐渐被淘汰,JDK1.7以后将字符串常量池和静态变量移入堆区,在JDK8以后,就不再有永久代说法,而是元空间(Meta Space)实现方法区。元空间和永久代最大的区别就是元空间不在JVM内存之中,而是使用本地内存,这样可以大大降低方法区的OOM。

    字符串常量池被移到Java堆的原因?
      方法区的垃圾回收只有在Full GC时才会发生,又因为String字符串创建频繁,放回堆中,使得能够及时回收。


    三、方法区大小设置

    1、永久代(JDK1.7)

    • -XX:PermSize : 永久代初始化内存,默认20.75M, -XX:PermSize=20m
    • -XX:MaxPermSize: 永久代最大内存,32位虚拟机默认64M, 64位默认82M,超出时会报java.lang.OutOfMemoryError:PermaGen space

    2、元空间(JDK1.8)

    • -XX:MetaspaceSize:元空间初始化内存,默认是21M
    • -XX:MaxMetaspaceSize:元空间最大内存,默认是-1无限制。虽然使用直接内存,但是当无法再为Metaspace分配内存时,将会抛出java.lang.OutOfMemoryError: Metaspace

    四、方法区OOM

    1、字符串常量池溢出导致OOM

      基于JDK1.6,字符串常量池在永久代之中,-XX:MaxPermSize=6m限制永久代最大内存为6M,执行以下代码,将会出现java.lang.OutOfMemoryError: PermaGen space

    public class PermGenOOMDemo {
        public static void main(String[] args) {
            Set<String> set = new HashSet<>();
            short i = 0;
            while (true) {
                set.add(String.valueOf(i++).intern());
            }
        }
    }

      JDK1.7以后字符串常量池进入Java堆,执行以上代码就不会出现OOM,而限制最大堆内存-Xmx之后,将同样出现OOM,只不过错误变成java.lang.OutOfMemoryError: Java heap space。

    2、类型数据加载导致OOM

      类型数据是方法区的主要组成部分,无论JDK1.6、JDK1.7或是1.8在设置永久代上限或元空间上限的情况下,都会出现OOM,如下代码,引入CGLib依赖,不断创建动态代理类放入方法区:

    • 当JVM参数为-XX:MaxPermSize=6m时,抛出异常java.lang.OutOfMemoryError: PermaGen space
    • 当JVM参数为—XX:MaxMetaspaceSize=6m时,抛出异常java.lang.OutOfMemoryError: Metaspace
        <dependency>
            <groupId>cglib</groupId>
            <artifactId>cglib</artifactId>
            <version>3.3.0</version>
        </dependency>
    public class MethodOOM {
        static class OOMObject {
    
        }
    
        public static void main(final String[] args) {
            while (true) {
                Enhancer enhancer = new Enhancer();
                enhancer.setSuperclass(OOMObject.class);
                enhancer.setUseCache(false);
                enhancer.setCallback(new MethodInterceptor() {
                    public Object intercept(Object o, Method method, Object[] objects, MethodProxy methodProxy) throws Throwable {
                        return methodProxy.invokeSuper(o, args);
                    }
                });
                enhancer.create();
            }
        }
    }

     

  • 相关阅读:
    c# CLR无法从 COM 上下文 0x51cd20 转换为 COM 上下文 0x51ce90
    sql语法
    学籍管理系统
    【Android进阶】Android调用WebService的实现
    【Android进阶】自定义控件实现底部扇形展开菜单效果
    华为上机题汇总----java
    卡片游戏(栈和队列)
    18岁生日
    循环多少次?
    Flappy bird源代码(略吊)
  • 原文地址:https://www.cnblogs.com/zhexuejun/p/15725496.html
Copyright © 2011-2022 走看看