java启动默认堆内存_JDK1.8 JVM运行时数据区域划分以及 堆空间内存分配(默认情况下)...

本文介绍了JDK1.8的JVM运行时数据区域,包括程序计数器、虚拟机栈、本地方法栈、堆和元数据区。堆空间在默认情况下分为老年代和年轻代,年轻代又细分为eden区和两个survivor区。元数据区替代了永久代,用于存储类信息等。通过示例代码展示了如何触发元数据区和堆的OOM异常。
摘要由CSDN通过智能技术生成

656b3821feca0fcb5a167be1524f44fe.png

一、JDK1.8 JVM运行时数据区域概览

75654b14e2aba6cf02b4bb3fc1564d1f.png

这里介绍的是JDK1.8 JVM运行时内存数据区域划分。1.8同1.7比,最大的差别就是:元数据区取代了永久代。元空间的本质和永久代类似,都是对JVM规范中方法区的实现。不过元空间与永久代之间最大的区别在于:元数据空间并不在虚拟机中,而是使用本地内存。

cf7fa0d9748ea08ff6575326ca62f66b.png

二、各区域介绍

1. 程序计数器

每个线程一块,指向当前线程正在执行的字节码代码的行号。如果当前线程执行的是native方法,则其值为null。

2. Java虚拟机栈

6349acd98c5bd1846420df67c1a9cb5c.png

线程私有,每个线程对应一个Java虚拟机栈,其生命周期与线程同进同退。每个Java方法在被调用的时候都会创建一个栈帧,并入栈。一旦完成调用,则出栈。所有的的栈帧都出栈后,线程也就完成了使命。

3. 本地方法栈

功能与Java虚拟机栈十分相同。区别在于,本地方法栈为虚拟机使用到的native方法服务。不多说。

4. 堆

c7bb7280fade624c38e0064a7c2d9213.png

堆是JVM内存占用最大,管理最复杂的一个区域。其唯一的用途就是存放对象实例:几乎所有的对象实例及数组都在对上进行分配。1.7后,字符串常量池从永久代中剥离出来,存放在堆中。堆有自己进一步的内存分块划分,按照GC分代收集角度的划分请参见上图。

4.1 堆空间内存分配(默认情况下)

2feb30e18a1f556f173d40a92f6fc1ae.png

老年代 : 三分之二的堆空间

年轻代 : 三分之一的堆空间

eden区: 8/10 的年轻代空间

survivor0 : 1/10 的年轻代空间

survivor1 : 1/10 的年轻代空间(from 区)

命令行上执行如下命令,查看所有默认的jvm参数

java -XX:+PrintFlagsFinal -version

输出:

fbe8a771c0e86511871ffe57b6080e5b.png

1b5b87bb5b06eee86ddd644b24e83e28.png

4.2 字符串常量池

JDK1.7 就开始“去永久代”的工作了。 1.7把字符串常量池从永久代中剥离出来,存放在堆空间中。

a. jvm参数配置

-XX:MaxPermSize=10m

-XX:PermSize=10m

-Xms100m

-Xmx100m

-XX:-UseGCOverheadLimit

b. 测试代码

public class StringOomMock {

public static void main(String[] args) {

try {

List list = new ArrayList();

for (int i = 0; ; i++) {

System.out.println(i);

list.add(String.valueOf("String" + i++).intern());

}

} catch (java.lang.Exception e) {

e.printStackTrace();

}

}

}

3ace1a9eda5af19b11adf55a1152b6eb.png

5. 元数据区

元数据区取代了1.7版本及以前的永久代。元数据区和永久代本质上都是方法区的实现。方法区存放虚拟机加载的类信息,静态变量,常量等数据。

元数据区OOM测试:

a. jvm参数配置

-XX:MetaspaceSize=8m

-XX:MaxMetaspaceSize=50m

b. 测试代码

借助cglib框架生成新类。

public class MetaSpaceOomMock {

public static void main(String[] args) {

ClassLoadingMXBean loadingBean = ManagementFactory.getClassLoadingMXBean();

while (true) {

Enhancer enhancer = new Enhancer();

enhancer.setSuperclass(MetaSpaceOomMock.class);

enhancer.setCallbackTypes(new Class[]{Dispatcher.class, MethodInterceptor.class});

enhancer.setCallbackFilter(new CallbackFilter() {

@Override

public int accept(Method method) {

return 1;

}

@Override

public boolean equals(Object obj) {

return super.equals(obj);

}

});

Class clazz = enhancer.createClass();

System.out.println(clazz.getName());

//显示数量信息(共加载过的类型数目,当前还有效的类型数目,已经被卸载的类型数目)

System.out.println("total: " + loadingBean.getTotalLoadedClassCount());

System.out.println("active: " + loadingBean.getLoadedClassCount());

System.out.println("unloaded: " + loadingBean.getUnloadedClassCount());

}

}

}

1b04b425bd8769919986357906084e6f.png

如果是1.7的jdk,那么报OOM的将是PermGen区域。

6. 直接内存

jdk1.4引入了NIO,它可以使用Native函数库直接分配堆外内存。

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值