[JVM教程与调优] 了解JVM 堆内存溢出以及非堆内存溢出

从图中我们可以看到,JVM的内存结构分为两大块。一块叫堆区,一块叫非堆区

堆区又分为两大块,一块Young,一块叫OldYoung区又分为Survivor区和Eden区。Survivor区我们又分为S0与S1。可以结合下图进行理解

JVM堆区

非堆区呢,是属于我们操作系统的本地内存。它是独立于我们堆区之外的。它在JDK1.8里面有一个新的名字,叫MetaspaceMetaspace里面还包含几个块,其中有一块就是CCS,还有一块是CodeCache。当然,在我们的Metaspace中还包含很多其他块,这里就不做扩展了。

接下来,我们来通过实战,来更加深入的理解JVM结构,以及出现JVM内存溢出的原因。

实战理解


我们通过spring.start快速来生成一个springboot项目。

快速实战

如图,我们快速的创建一个springboot项目,并将其下载下来。

这里我使用Eclipse,小伙伴们也可以使用IDEA或者其他开发工具也是可以的。

这里我们使用的是SpringBoot工程,如果有的小伙伴对SpringBoot还不太熟悉的,可以上网找一些教程先学习了解一下。

堆内存溢出演示


那么我们如何来构建一个堆内存溢出呢?其实很简单,我们只要定义一个List对象,然后通过一个循环不停的往List里面塞对象。因为只要Controller不被回收,那么它里面的成员变量也是不会被回收的。这样就会导致List里面的对象越来越多,占用的内存越来越大,最后就把我们的内存撑爆了。

创建User对象

这里我们先创建一个User对象。

/**

*

Title: User

Description: 

* @author Coder编程

* @date 2020年3月29日

*/

@Data

@AllArgsConstructor

@NoArgsConstructor

public class User {

private int id;

private String name;

}

这里面@Data@AllArgsConstructor@NoArgsConstructor用的是lombok注解。不会使用的小伙伴,可以在网上查找相关资料学习一下。

创建Controller对象

接下来我们来创建一个Controller来不停的往List集合中塞对象。

/**

*

Title: MemoryController

Description: 

* @author Coder编程

* @date 2020年3月29日

*/

@RestController

public class MemoryController {

private List  userList = new ArrayList();

/**

* -Xmx32M -Xms32M

* */

@GetMapping(“/heap”)

public String heap() {

int i=0;

while(true) {

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

}

}

}

为了更快达到我们的效果,我们来设置两个参数。

-Xmx32M -Xms32M

一个最大内存,一个最小内存。我们的堆就只有32M,这样就很容易溢出。

访问测试

启动时候设置内存参数。

设置内存参数1

设置内存参数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,小伙伴如果有遇到,可尝试一下是否能够解决
自我介绍一下,小编13年上海交大毕业,曾经在小公司待过,也去过华为、OPPO等大厂,18年进入阿里一直到现在。

深知大多数Java工程师,想要提升技能,往往是自己摸索成长或者是报班学习,但对于培训机构动则几千的学费,着实压力不小。自己不成体系的自学效果低效又漫长,而且极易碰到天花板技术停滞不前!

因此收集整理了一份《2024年Java开发全套学习资料》,初衷也很简单,就是希望能够帮助到想自学提升又不知道该从何学起的朋友,同时减轻大家的负担。img

既有适合小白学习的零基础资料,也有适合3年以上经验的小伙伴深入学习提升的进阶课程,基本涵盖了95%以上Java开发知识点,真正体系化!

由于文件比较大,这里只是将部分目录截图出来,每个节点里面都包含大厂面经、学习笔记、源码讲义、实战项目、讲解视频,并且会持续更新!

如果你觉得这些内容对你有帮助,可以扫码获取!!(备注Java获取)

img

Spring全套教学资料

Spring是Java程序员的《葵花宝典》,其中提供的各种大招,能简化我们的开发,大大提升开发效率!目前99%的公司使用了Spring,大家可以去各大招聘网站看一下,Spring算是必备技能,所以一定要掌握。

目录:

部分内容:

Spring源码

  • 第一部分 Spring 概述
  • 第二部分 核心思想
  • 第三部分 手写实现 IoC 和 AOP(自定义Spring框架)
  • 第四部分 Spring IOC 高级应用
    基础特性
    高级特性
  • 第五部分 Spring IOC源码深度剖析
    设计优雅
    设计模式
    注意:原则、方法和技巧
  • 第六部分 Spring AOP 应用
    声明事务控制
  • 第七部分 Spring AOP源码深度剖析
    必要的笔记、必要的图、通俗易懂的语言化解知识难点

脚手框架:SpringBoot技术

它的目标是简化Spring应用和服务的创建、开发与部署,简化了配置文件,使用嵌入式web服务器,含有诸多开箱即用的微服务功能,可以和spring cloud联合部署。

Spring Boot的核心思想是约定大于配置,应用只需要很少的配置即可,简化了应用开发模式。

  • SpringBoot入门
  • 配置文件
  • 日志
  • Web开发
  • Docker
  • SpringBoot与数据访问
  • 启动配置原理
  • 自定义starter

微服务架构:Spring Cloud Alibaba

同 Spring Cloud 一样,Spring Cloud Alibaba 也是一套微服务解决方案,包含开发分布式应用微服务的必需组件,方便开发者通过 Spring Cloud 编程模型轻松使用这些组件来开发分布式应用服务。

  • 微服务架构介绍
  • Spring Cloud Alibaba介绍
  • 微服务环境搭建
  • 服务治理
  • 服务容错
  • 服务网关
  • 链路追踪
  • ZipKin集成及数据持久化
  • 消息驱动
  • 短信服务
  • Nacos Confifig—服务配置
  • Seata—分布式事务
  • Dubbo—rpc通信

Spring MVC

目录:

部分内容:

《互联网大厂面试真题解析、进阶开发核心学习笔记、全套讲解视频、实战项目源码讲义》点击传送门即可获取!
Cloud Alibaba介绍

  • 微服务环境搭建
  • 服务治理
  • 服务容错
  • 服务网关
  • 链路追踪
  • ZipKin集成及数据持久化
  • 消息驱动
  • 短信服务
  • Nacos Confifig—服务配置
  • Seata—分布式事务
  • Dubbo—rpc通信

[外链图片转存中…(img-6W9lMmXa-1713512263436)]

[外链图片转存中…(img-fNJBD1CK-1713512263436)]

Spring MVC

目录:

[外链图片转存中…(img-0J0TQFa8-1713512263437)]

[外链图片转存中…(img-6BGiCT9U-1713512263438)]

[外链图片转存中…(img-qtBPZ878-1713512263438)]

部分内容:

[外链图片转存中…(img-kSiB4OuY-1713512263439)]

[外链图片转存中…(img-58wBpxAO-1713512263439)]

《互联网大厂面试真题解析、进阶开发核心学习笔记、全套讲解视频、实战项目源码讲义》点击传送门即可获取!

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

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值