JVM实战篇

内存调优

内存溢出和内存泄漏

内存泄漏:在java中如果不再使用一个对象,但是该对象依然在GC ROOT的引用链上,这个对象就不会被垃圾回收器回收。

内存泄漏绝大多数情况都是由堆内存泄漏引起的,所以后续没有特别说明则讨论的都是堆内存泄漏。

少量的内存泄漏是可以容忍的,但是如果发生持续的内存泄漏,就像滚雪球一样越滚越大,不管有多大的内存迟早都会被消耗完,最终导致的结果就是内存溢出但是产生内存溢出的原因并不是只有内存泄漏。 

常见的内存泄漏就是在大型的java后端应用中,在处理用户的请求之后,没有及时将用户的数据删除。随着用户请求数量越来越多,内存泄漏的对象占满了堆内存最终导致内存溢出。

分布式任务调度系统如Elastic-job、Quartz等进行任务调度是,被调度的Java应用在调度任务结束中出现了内存泄漏,最终导致多次调度之后内存溢出。

解决内存溢出的方法

有四个步骤:发现 诊断 修复 验证

发现问题

TOP命令

VisualVM

 Arthas

 使用arthas tunnel管理所有的需要监控的程序

步骤:

 Prometheus+Grafana

 堆内存状况的对比

 正常就是上下上下,不正常就是一直增长

产生内存溢出的原因一:代码中的内存泄漏

1.equals()和hashcode()

 在定义新类的时候没有重写正确的equals()和hashcode()方法。在时候HashMap,如果使用这个类对象作为key,HashMap在判断Key是否已经存在时,会使用这两个方法,如果重写不正确,就会导致相同的数据被保存多份。

 

解决方案

1.在定义新实体的时候,始终重写epuals()和hashcode()方法、

2.重写时一定要确定使用了唯一的标识去区分不同的对象,比如用户的id等。

3.hashmap使用时尽量使用编号id等数据作为key,不要将整个实体类对象作为key存放

2.内部类引用外部类

(1)非静态的内部类默认会持有外部类,尽管代码上不再使用外部类,所以如果有地方引用了这个非静态内部类,会导致外部类也被引用,垃圾回收时无法回收这个外部类。

解决方案:把内部类改成静态的

(2)匿名内部类对象如果在非静态方法中被创建,会持有调用者对象,垃圾回收时无法回收调用者。 

解决方案:吧那个方法变成静态的

 

3.ThreadLocal的使用

如果仅仅使用手动创建的线程,就算没有调用ThreadLocal的remove方法清理数据,也不会产生内存泄漏。因为当线程被回收时,ThreadLocal也照样会被回收。但是如果使用线程池就不一定了。

解决方案:线程方法执行完,一定要调用ThreadLocal中的remove方法清理对象。

4.String的intern方法

JDK6中字符串常量池位于堆内存中的Perm Gen永久代中,如果不同字符串的intern方法被大量调用,字符串常量池会不停的变大超过永久代上限之后就会产生内存溢出问题。

解决方案:

1.注意代码逻辑,尽量不要将随机生成的字符串加入字符串常量池

2.增加永久代空间的大小,根据实际的测试/估算结果进行设置-XX:MaxPermSize=256M

5.通过静态字段保存对象

如果大量的数据在静态变量中被长期引用,数据就不会被释放,如果这些数据不再使用,就成为了内存泄漏。

解决方案:

(1)尽量减少将对象长时间的保存在静态变量中,如果不再使用,必须将对象删除(比如在集合中)或者将静态变量设置成null

(2)使用单例模式时,尽量使用懒加载,而不是立即加载。

(3)Spring的Bean中不要长期存放大对象,如果是缓存用于提升性能,尽量设置过期时间定期失效。

6.资源没有正常关闭

连接和流这些资源会占用内存,如果使用完之后没有关闭,这部分内存不一定会出现内存泄漏(有可能会),但是会导致close方法不被执行

解决方案:

1.为了防止出现这类的资源对象泄漏问题,必须在finally块中关闭不再使用的资源。

2.从Java7开始,使用try-with-resources语法可以用于自动关闭资源

内存溢出原因二:并发请求问题

模拟并发请求

Apache Jmeter 

它支持插件扩展,生成多样化的测试结果

诊断-内存快照

当堆内存溢出的时候,需要在堆内存溢出的时候将整个堆内存保存下来,生成内存快照文件。

使用MAT打开hprof文件,并选择内存泄漏检测功能,MAT会自行根据内存快照中保存的数据分析内存泄漏的根源。

 

MAT内存泄漏检测的原理---支配树

MAT提供了称为支配树的对象图。支配树展示的是对象实例间的支配关系。在对象引用图中,所有指向对象B的路径都经过对象A则认为对象A支配对象B

浅堆:支配树中对象本身占用的空间(自己的节点)

深堆:支配树中对象的子树就是所有被该对象支配的内容,这些内容组成了对象的深堆,也叫保留集。深堆的大小表示该对象如果可以被回收,能释放多大的内存空间。(自己的节点+子树的节点)

导出运行中系统的内存快照并进行分析

分析超大堆的内存快照

 

修复问题

修复代码中的问题

问题总共可以分为三类

案例1-分页查询文章接口的内存溢出

 解决思路:

1.服务出现OOM内存溢出时,生成内存快照

2.使用MAT分析内存快照,找到内存溢出的对象

3.尝试在开发环境中重现问题,分析代码中问题产生的原因

4.修改代码

5.测试并验证结果

上面案例的解决方案

案例2:mybatis导致的内存溢出

案例3:导出大文件内存溢出

k8s容器

 

 案例4:ThreadLocal使用时占用了大量内存

 

案例5:文章内容审核接口的内存问题

设计1:

 

用MQ是最好的

 诊断方法二 在线定位问题

在线定位问题的步骤

GC调优

就是对垃圾回收进行调优。GC调优的主要目标是避免由垃圾回收引起程序性能下降。

GC调优分为三部分:
1.通过JVM参数设置。

2.特定垃圾回收器的JVM参数设置

3.解决由频繁的FULLGC引起的程序性能问题。

GC调优的核心指标

1.吞吐量

垃圾回收吞吐量:指CPU用于执行用户代码的时间与CPU总执行时间的比值。吞吐量=执行用户代码的时间/(执行用户代码的时间+GC时间)。吞吐量越高垃圾回收效率就越高。允许更多的CPU时间去处理用户的业务,相应的业务吞吐量也就越高。

2.延迟

延迟是指从用户发起一个请求到收到响应这其中经历的时间。

3.内存的使用量

内存使用量指的是Java应用占系统内存的最大值,一般tongguoJVM参数调整,在满足上述两个指标的前提下,这个值越小越好。

GC调优的方法

和内存调优差不多

发现问题

jstat工具

 visualvm

Prometheus+Grafana

GC日志文件

GC Viewer

GCeasy

常见的GC模式

持续的FULLGC是主要解决的问题

诊断问题

选择前三个为主。

优化基础JVM参数

减少对象的产生

就是根据内存调优的方案进行,修改代码

更换垃圾回收器

优化垃圾回收器参数

 

解决思路

实现 GC调优和内存调优

核心流程

性能调优

性能调优共分为4个步骤,其中修复部分要具体问题具体分析且处理方式各不相同

本章着重学习发现问题和诊断问题的方法,目标是准确定位到性能问题的根源

性能调优解决的问题

应用程序在运行过程中经常会出现性能问题,比较常见的性能问题现象是:

1.通过top命令查看CPU占用率高,接近100甚至多核CPU下超过100都是有可能的

2.请求单个服务处理时间特别长,多服务使用skywalking等监控系统来判断是哪个环节性能低下。

3.程序启动之后运行正常,但是在运行一段时间之后无法处理任务的请求(内存和GC正常)

性能调优的核心方法

线程转储的查看方式

 解决CPU占用率高的问题

案例2:接口响应的时间很长的问题

Arthas的trace命令

watch命令

案例3:定位偏底层的性能问题

案例4:线程被耗尽问题

更精细化的性能测试

JIT对程序性能的影响

java程序在运行过程中,JIT即时编译器会实时对代码进行性能优化,所以仅凭少量的测试是无法真实反应运行系统最终给用户提供的性能。如下图,随着执行次数的增加,程序性能会逐渐优化

jmh在使用过程的问题

案例:日期格式化方法性能测试

案例实战部分

问题:小李的项目中有一个获得用户信息的接口性能比较差,他希望能对这个接口在代码中进行彻底的优化,提升性能。

总结

总结

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.mzph.cn/news/750959.shtml

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

Linux-centos如何搭建yum源仓库

1.本地搭建(无需连接外网) 1.1检查网络配置,及网络连接 打开虚拟机,点击【编辑——虚拟网络编辑器】 点击【仅主机模式】查看子网段是否和局内IP匹配 进入局内,查看网络IP是否在你上述设置的网段内,如果不…

Chapter 13 Techniques of Design-Oriented Analysis: The Feedback Theorem

Chapter 13 Techniques of Design-Oriented Analysis: The Feedback Theorem 从这一章开始讲负反馈Control系统和小信号建模. 13.2 The Feedback Theorem 首先介绍 Middlebrook’s Feedback Theorem 考虑下面负反馈系统 传输函数 Guo/ui G ( s ) u o u i G ∞ T 1 T G…

1.实用Qt:解决绘制圆角边框时,圆角锯齿问题

目录 问题描述 解决方案 方案1: 方案2: 结果示意图 问题描述 做UI的时候,我们很多时候需要给绘制一个圆角边框,初识Qt绘制的童鞋,可能绘制出来的圆角边框很是锯齿,而且粗细不均匀,如下图&…

Vue | 使用 ECharts 绘制折线图

目录 一、安装和引入 ECharts 二、使用 ECharts 2.1 新增 div 盒子 2.2 编写画图函数 2.3 完整代码结构 三、各种小问题 3.1 函数调用问题 3.2 数据格式问题 3.3 坐标轴标签问题 3.4 间隔显示标签 参考博客:Vue —— ECharts实现折线图 本文是在上…

JVM学习-JVM简介以及其内部结构

目录 1.什么是JVM 2.JVM、JRE、JDK、JavaSE、JavaEE之间的联系 3.JVM的内部结构 4.各部分的作用 4.1 类加载器 4.2 方法区 4.3 堆 ​编辑 4.4 虚拟机栈 4.5 程序计数器 4.6 本地方法栈 4.7 解释器和JIT即时编译器 4.9 GC垃圾回收 5.拓展 5.1一些可能会遇到的问…

opencv安装(C++)并配置vs

准备工作: 1.opencv安装包(此教程使用4.9) 2.visual studio(此教程使用vs2019) opencv安装: 1、下载opencv: 1.1 官网下载:Releases - OpenCV 1.2 百度网盘:链接:https://pan.baidu.com/s/1NpEoFjbbyQJtFD…

CTF-希尔加解密

对于希尔加解密很多writeup都说用在线工具,所以研究了一下,写了一个方便的加解密python代码,根据给定的字母表及私钥字符串,尝试不同纬度不同重叠的加密矩阵输出加解密结果。运行效果如下: 代码文件Hill希尔加解密_fi…

8:00面试,8:06就出来了,问的问题有点变态。。。

从小厂出来,没想到在另一家公司又寄了。 到这家公司开始上班,加班是每天必不可少的,看在钱给的比较多的份上,就不太计较了。没想到9月一纸通知,所有人不准加班,加班费不仅没有了,薪资还要降40%…

matlab simulink 工业现场漆包机烘炉模糊PID控制算法

1、内容简介 略 70-可以交流、咨询、答疑 2、内容说明 matlab simulink 工业现场漆包机烘炉模糊PID控制算法 第五章 控制算法仿真与分析 在第二章系统分析结合工业现场漆包机烘炉数学模型详细分析模糊PID控制算法以及解耦控制算法的优缺点、实现原理,为本章算法…

第8节课------列生成与分支定价法

列生成与分支定价法 1. 列生成 通过迭代的方式来构建和解决一个“受限”的主问题,并通过解决一系列相关的子问题来动态地生成新的变量(或称列),这些新变量有可能改进主问题的解。 原问题新添加的这一列可以通过对偶问题的行来计算。 2. 列生成的详细说明 对于原来的…

【TB作品】数字电压表设计,MSP430,电压表,Proteus仿真,分档位

文章目录 题目要求第1版方案第2版方案代码介绍获取代码和仿真 题目要求 33、数字电压表设计 基本要求: 了解电压表的工作原理。 对0~2.5V的模拟电压进行循环采集 采集的数据进行A/D转换后用LCD屏显示测量值,显示精度0.0001 用键盘选择测量范围(分档) 超过测量范围时指示灯持续…

基于springboot+vue实现员工信息管理系统项目【项目源码+论文说明】

基于springbootvue实现员工信息管理系统演示 引言 随着计算机技术的飞速发展,计算机在企业管理中应用的普及,利用计算机在实现企业人事档案的管理势在必行。当今社会正快速向信息化社会前进,信息自动化的作用也越来越大。从而使我们从繁杂的…

上位机图像处理和嵌入式模块部署(qmacvisual跳转语句)

【 声明:版权所有,欢迎转载,请勿用于商业用途。 联系信箱:feixiaoxing 163.com】 前面我们依次讨论了判断语句和结束判断语句,今天我们继续学习跳转语句。从功能上面来说,跳转语句和判断语句较为类似。不过…

时间序列预测的零样本学习是未来还是炒作:TimeGPT和TiDE的综合比较

最近时间序列预测预测领域的最新进展受到了各个领域(包括文本、图像和语音)成功开发基础模型的影响,例如文本(如ChatGPT)、文本到图像(如Midjourney)和文本到语音(如Eleven Labs&…

【JS】html字符转义

需求 将html转为字符串将html字符串转义&#xff0c;比如<div>转为<div> 码 /*** html标签字符转义* param {Stirng} str 要转换的html字符* returns String 返回转义的html字符串*/ const elToStr str > str.replaceAll(<, <).replaceAll(>, >)…

ChatGPT有身体了,能走向千家万户吗?

3月13日&#xff0c;OpenAI投资的人形机器人——Figure 01迎来重磅更新&#xff0c;接入最新版ChatGPT后&#xff0c;它能和人交流描述眼前看到的事物。在视频中&#xff0c;Figure 的人形机器人&#xff0c;可以完全与人类流畅对话&#xff0c;还能理解人的自然语言指令进行抓…

搞懂分布式技术:缓存更新的套路

目录 缓存更新的套路 Cache Aside Pattern Read/Write Through Pattern Read Through Write Through Write Behind Caching Pattern 再多唠叨一些 缓存更新的套路 看到好些人在写更新缓存数据代码时&#xff0c;先删除缓存&#xff0c;然后再更新数据库&#xff0c;而后续的操作…

代码随想录 -- 回溯算法

文章目录 回溯算法理论什么是回溯法回溯法的效率回溯法解决的问题理解回溯法回溯法模板 组合问题I描述题解优化 组合总和III描述题解 电话号码的字母组合描述题解 组合总和描述题解 组合总和II描述题解 分割回文串描述题解 复原IP地址描述题解 子集描述题解 子集II描述题解 递增…

shallowReactive浅层式响应对象

一、 reactive 和ref 都是深层响应式对象: 就是不管对象有多少层&#xff0c;修改此对象任一属性都会响应式处理 shallowReactive 和shallowRef 浅层响应式对象: 只会修改第一层对象&#xff0c;修改此对象第一层属性&#xff0c;视图会有同步变化&#xff0c;非第一层&#xf…

WorkPlus Meet局域网视频会议软件的领先解决方案

局域网视频会议软件在现代企业中发挥着重要的作用&#xff0c;而在众多选项中&#xff0c;为何选择WorkPlus Meet作为局域网视频会议软件&#xff1f; 选择局域网视频会议软件时需要考虑到企业的需求。WorkPlus Meet提供了稳定、高效的局域网视频会议功能&#xff0c;能够满足…