java8中使用Metaspace就不会出现OOM吗?

来源:互联网 发布:知行的初始密码 编辑:程序博客网 时间:2024/06/05 07:44

前言:在java8中,Metaspace的出现,使我们现在不会再遇到java.lang.OutOfMemoryError: PermGen问题,但是我们要记住,这个新特性并不会使类加载导致的内存泄露就此消失。

(一)Metaspace的简单介绍
(1)内存模型:大部分类元数据都在本地内存分配,用于描述类元数据的“klasses“已经被移除。
(2) 容量:默认情况下只受本地内存限制,但是我们可以限制。

(二)接下来看看Metaspace会不会出现OOM
我们可以通过动态代理来生成大量的代理类信息填充元数据区。代码如下

public interface MetaspaceFacade {    void method(String input);}
public class MetaspaceFacadeImpl implements MetaspaceFacade {    public void method(String name) {    }}
public class MetaspaceFacadeInvocationHandler implements InvocationHandler {    private Object classAImpl;    public MetaspaceFacadeInvocationHandler(Object impl) {       this.classAImpl = impl;    }    @Override    public Object invoke(Object proxy, Method method, Object[] args) throws Throwable {       return method.invoke(classAImpl, args);    }}
public class ClassMetadataLeakSimulator {    private static Map<String, MetaspaceFacade> classLeakingMap = new HashMap<String, MetaspaceFacade>();    private final static int NB_ITERATIONS_DEFAULT = 50000;    /**     * @param args     */    public static void main(String[] args) {        System.out.println("Class metadata leak simulator");        int nbIterations = NB_ITERATIONS_DEFAULT;        try {            for (int i = 0; i < nbIterations; i++) {                String fictiousClassloaderJAR = "file:" + i + ".jar";                URL[] fictiousClassloaderURL = new URL[] { new URL(fictiousClassloaderJAR) };                URLClassLoader newClassLoader = new URLClassLoader(fictiousClassloaderURL);                MetaspaceFacade t = (MetaspaceFacade) Proxy.newProxyInstance(newClassLoader,                        new Class<?>[] { MetaspaceFacade.class },                        new MetaspaceFacadeInvocationHandler(new MetaspaceFacadeImpl()));                classLeakingMap.put(fictiousClassloaderJAR, t);            }        }         catch (Throwable any) {            System.out.println("ERROR: " + any);        }        System.out.println("Done!");    }}

1,首先不进行任何设置(需要保证堆大小),运行得到下面的截图
这里写图片描述
可以看出,Metaspace进行了动态扩展,本地内存达到了300多M,加载超过5万个类后还没有出现OOM事件。接下来我们限定一下

2,虚拟机参数配置如下:-Xmx2g -Xms2g -Xmn1g -XX:+PrintGCDetails -XX:MaxMetaspaceSize=128m,运行得到下面的截图
这里写图片描述
可以看出,此时Metaspace被耗尽,将会抛出OOM事件。这里要注意MaxPermSize的区别,MaxMetaspaceSize并不会在jvm启动的时候分配一块这么大的内存出来,而MaxPermSize是会分配一块这么大的内存的。

那么问题来了,如果我们不设置MaxMetaspaceSize,那么我们能用到系统的全部内存吗??
metaspace很可能因为被无止境使用而被OS Kill,所以一般还是设置比较好。

原创粉丝点击