Java最新[JVM教程与调优] 了解JVM 堆内存溢出以及非堆内存溢出,1-3年的Java开发工程师看过来

最后希望可以帮助到大家!

千千万万要记得:多刷题!!多刷题!!

之前算法是我的硬伤,后面硬啃了好长一段时间才补回来,算法才是程序员的灵魂!!!!

篇幅有限,以下只能截图分享部分的资源!!

(1)多线程(这里以多线程为代表,其实整理了一本JAVA核心架构笔记集)

image

(2)刷的算法题(还有左神的算法笔记)

image

(3)面经+真题解析+对应的相关笔记(很全面)

image

(4)视频学习(部分)

ps:当你觉得学不进或者累了的时候,视频是个不错的选择

在这里,最后只一句话:祝大家offer拿到手软!!

本文已被CODING开源项目:【一线大厂Java面试题解析+核心总结学习笔记+最新讲解视频+实战项目源码】收录

需要这份系统化的资料的朋友,可以点击这里获取

设置内存参数2

记得选中我们的Arguments,在JVM 参数中,将我们的值设置进去。最后点击Run运行起来。

然后我们在浏览器中请求:

http://localhost:8080/heap

我们再观察控制台打印:

打印结果

通过打印结果,我们可以看到堆内存溢出了。

注意:

这里我们测试的时候可以很简单的看出在哪里出现的问题,但是在实际生产环境中并没有那么简单,因此我们需要借助工具,来定位这些问题。后续我们来介绍一下。

非堆内存溢出演示


接下来我们来演示一下非堆内存溢出,我们继续沿用上方代码。

非堆内存主要是MataSpace,那么我们如何构建一个非堆内存溢出呢?

我们知道MataSpace主要存一些class,filed,method等这些东西。

因此我们继续创建一个List集合,不断的往集合里面塞class。只要List不被回收,那么它里面的class也不会被回收。不停的往里面加之后,就会造成溢出。也就是我们的MataSpace溢出了。

如何来动态生成一些class呢?其实是有很多工具的,比如说:asm

引入asm工具包

这里我们引入asm jar包。

asm

asm

3.3.1

动态生成类文件

还需要创建动态生成的类文件,这里我们就不做扩展介绍,有兴趣的小伙伴可以自行到网上查阅。

/**

*

Title: Metaspace

Description: https://blog.csdn.net/bolg_hero/article/details/78189621

* 继承ClassLoader是为了方便调用defineClass方法,因为该方法的定义为protected

* @author Coder编程

* @date 2020年3月29日

*/

public class Metaspace extends ClassLoader {

public static List<Class<?>> createClasses() {

// 类持有

List<Class<?>> classes = new ArrayList

// 循环1000w次生成1000w个不同的类。

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

ClassWriter cw = new ClassWriter(0);

// 定义一个类名称为Class{i},它的访问域为public,父类为java.lang.Object,不实现任何接口

cw.visit(Opcodes.V1_1, Opcodes.ACC_PUBLIC, “Class” + i, null,

“java/lang/Object”, null);

// 定义构造函数方法

MethodVisitor mw = cw.visitMethod(Opcodes.ACC_PUBLIC, “”,

“()V”, null, null);

// 第一个指令为加载this

mw.visitVarInsn(Opcodes.ALOAD, 0);

// 第二个指令为调用父类Object的构造函数

mw.visitMethodInsn(Opcodes.INVOKESPECIAL, “java/lang/Object”,

“”, “()V”);

// 第三条指令为return

mw.visitInsn(Opcodes.RETURN);

mw.visitMaxs(1, 1);

mw.visitEnd();

Metaspace test = new Metaspace();

byte[] code = cw.toByteArray();

// 定义类

Class<?> exampleClass = test.defineClass(“Class” + i, code, 0, code.length);

classes.add(exampleClass);

}

return classes;

}

}

创建Controller

接下来我们再原Controller新增一个方法nonheap

/**

*

Title: MemoryController

Description: 

* @author Coder编程

* @date 2020年3月29日

*/

@RestController

public class MemoryController {

private List  userList = new ArrayList();

private List<Class<?>>  classList = new ArrayList

/**

* -Xmx32M -Xms32M

* */

@GetMapping(“/heap”)

public String heap() {

int i=0;

while(true) {

userList.add(new User(i++, UUID.randomUUID().toString()));

}

}

/**

* -XX:MetaspaceSize=32M -XX:MaxMetaspaceSize=32M

* */

@GetMapping(“/nonheap”)

public String nonheap() {

while(true) {

classList.addAll(Metaspace.createClasses());

}

}

}

访问测试

这里我们同样在启动的时候也要设置Mataspace的值大小。

-XX:MetaspaceSize=32M -XX:MaxMetaspaceSize=32M

设置启动参数

接着我们在浏览器中访问地址:localhost:8080/nonheap

以上我们就完成了对堆内存溢出以及非堆内存溢出的演示。

小插曲

在测试非堆内存溢出的时候,出现了另外一个错误。

java.lang.IncompatibleClassChangeError: Found interface org.objectweb.asm.MethodVisitor, but class was expected

这个异常另外写在java.lang.IncompatibleClassChangeError,小伙伴如果有遇到,可尝试一下是否能够解决

如何查看线上堆内存溢出以及非堆内存溢出


我们主要查看线上的内存映像文件来查看到底是哪里发生了内存溢出。

发生内存溢出的主要原因:

1.内存发生泄漏

2.内存分配不足

假如发生内存泄漏的话,我们就需要找到是哪个地方发生了内存泄漏,一直占用内存没有释放。

下面我们来看一下如何来导出我们的内存映像文件。

主要有两种方式。

1.内存溢出自动导出

-XX:+HeapDumpOnOutOfMemoryError

-XX:HeapDumpPath=./

第一个参数表示:当发生内存溢出的时候,将内存溢出文件Dump出来。

第二个参数表示:Dump出来的文件存放的目录。

2.使用jmap命令手动导出

如果我们使用第一种命令,在发送内存溢出的时候再去导出,可能就有点晚了。我们可以等程序运行起来一段时间后,就可以使用jmap命令导出来进行分析。

演示内存溢出自动导出

我们需要用到两个命令参数。

-XX:+HeapDumpOnOutOfMemoryError

-XX:HeapDumpPath=./

自动导出命令参数

我们接着运行项目,访问:localhost:8080/heap

查看一下打印结果。

打印结果

可以看到,当发生了内存溢出后。输出了一个java_pid3972.hprof的文件。

在当前项目的当前文件中,我们就可以找到该文件。

演示jmap命令

option:-heap,-clstats,-dump:,-F

jmap命令

导出内存映像命令

参数都是什么意思呢?

live:只导出存活的对象,如果没有指定,则全部导出

format:导出文件的格式

file:导入的文件

我们刚才的程序还没有关闭,我们来看下程序的pid是多少。

输入:jps -l

查看pid

我们将其文件导入到桌面中来,输入命令

jmap -dump:format=b,file=heap.hprof 3972

最后

由于细节内容实在太多了,为了不影响文章的观赏性,只截出了一部分知识点大致的介绍一下,每个小节点里面都有更细化的内容!

小编准备了一份Java进阶学习路线图(Xmind)以及来年金三银四必备的一份《Java面试必备指南》

本文已被CODING开源项目:【一线大厂Java面试题解析+核心总结学习笔记+最新讲解视频+实战项目源码】收录

需要这份系统化的资料的朋友,可以点击这里获取

jmap -dump:format=b,file=heap.hprof 3972

最后

由于细节内容实在太多了,为了不影响文章的观赏性,只截出了一部分知识点大致的介绍一下,每个小节点里面都有更细化的内容!

[外链图片转存中…(img-OgtGDhyO-1715415766297)]

小编准备了一份Java进阶学习路线图(Xmind)以及来年金三银四必备的一份《Java面试必备指南》

[外链图片转存中…(img-4o34FsSv-1715415766298)]

本文已被CODING开源项目:【一线大厂Java面试题解析+核心总结学习笔记+最新讲解视频+实战项目源码】收录

需要这份系统化的资料的朋友,可以点击这里获取

  • 25
    点赞
  • 19
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值