最后希望可以帮助到大家!
千千万万要记得:多刷题!!多刷题!!
之前算法是我的硬伤,后面硬啃了好长一段时间才补回来,算法才是程序员的灵魂!!!!
篇幅有限,以下只能截图分享部分的资源!!
(1)多线程(这里以多线程为代表,其实整理了一本JAVA核心架构笔记集)
(2)刷的算法题(还有左神的算法笔记)
(3)面经+真题解析+对应的相关笔记(很全面)
(4)视频学习(部分)
ps:当你觉得学不进或者累了的时候,视频是个不错的选择
在这里,最后只一句话:祝大家offer拿到手软!!
设置内存参数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面试必备指南》
令
jmap -dump:format=b,file=heap.hprof 3972
最后
由于细节内容实在太多了,为了不影响文章的观赏性,只截出了一部分知识点大致的介绍一下,每个小节点里面都有更细化的内容!
[外链图片转存中…(img-OgtGDhyO-1715415766297)]
小编准备了一份Java进阶学习路线图(Xmind)以及来年金三银四必备的一份《Java面试必备指南》
[外链图片转存中…(img-4o34FsSv-1715415766298)]