谷粒商城:性能压测JVM堆区

目录

Kit

Apache JMeter

VisualVM

堆内存

jvm内存模型

垃圾回收(Garbage Collection, GC)

新对象分配内存

GC步骤

MinorGC

性能优化

影响因素

优化

nginx动静分离

优化三级分类获取

Jvm参数配置堆区

测试


Kit

Apache JMeter

压力测试,模拟大量用户并发访问

VisualVM

监控、分析和调优 Java 应用程序的性能

VisualGC 插件:监控垃圾回收


堆内存

jvm内存模型

    堆区:

    所有对象实例和数组都存储在堆区,堆区是线程共享的,所有线程都可以访问堆中的对象。

    堆区结构:

    • 年轻代(Young Generation)

      • Eden 区:新创建的对象首先分配在这里。

      • Survivor 区:分为 From 和 To 区,存放经过一次 GC 后仍然存活的对象。

    • 老年代(Old Generation):存放长期存活的对象。

    • 元空间(Metaspace):存放类的元数据(如类定义、方法信息等)。

    垃圾回收(Garbage Collection, GC)

    新对象分配内存

    1. 新创建的对象首先分配在 Eden 区
    2. 当 Eden 区满时,触发 Minor GC
    3. MinorGC后,若Eden区放得下,则存入Eden区
    4. 若MinorGC后,Eden空间仍然不足,则为大对象,直接分配至Old区
    5. 若Old区空间不足,触发FullGC,对整个堆内存进行垃圾回收。

    GC步骤

    (1) 标记(Marking)

    • 遍历所有对象,标记哪些对象是存活的(被引用的),哪些是垃圾(未被引用)。

    • 从 GC Roots(如线程栈、静态变量等)开始,递归标记所有可达对象。

    (2) 清除(Sweeping)

    • 清除未被标记的垃圾对象,释放它们占用的内存空间。

    • 清除后,内存可能会产生碎片。

    (3) 压缩(Compacting)

    • 将存活的对象移动到内存的一端,整理出连续的内存空间,减少内存碎片。

    • 这一步不是所有 GC 算法都会执行(如 CMS 就不压缩)。

    MinorGC

    1. 标记 Eden 区和 From 区(当前活动的 Survivor 区)中的存活对象。
    2. 将存活的对象复制到 To 区(另一个 Survivor 区)。
    3. 清空 Eden 区和 From 区,From和To角色互换

    角色交换

    • 在下次 Minor GC 时,原来的 To 区 变为新的 From 区,而原来的 From 区 变为新的 To 区

    • 这种交换确保了每次 Minor GC 都有一个空的 Survivor 区用于存放存活对象。 

    对象晋升: 

    • 如果对象在 Survivor 区中经历了多次 Minor GC(默认是 15 次,可以通过 -XX:MaxTenuringThreshold 参数调整),它会被晋升到 老年代

     为什么需要两个 Survivor 区?

    • 减少内存碎片:通过复制算法,将存活对象从一个 Survivor 区复制到另一个 Survivor 区,可以整理内存,减少碎片。

    • 提高垃圾回收效率:每次 Minor GC 只需要处理 Eden 区和其中一个 Survivor 区,而不是整个年轻代。


    性能优化

    影响因素

    中间件

    • client - > nginx - > 网关 - > server        中间件越多,网络IO交互越多,性能损耗越大

    业务

    •  DB(MySQl优化)
    • 模板渲染(开启缓存)
    • 静态资源(动静分离)

    优化

    1.日志:仅报错

    logging:level:com.elysia.gulimall.product: error

    2. MySQL加索引

    3.thtmeleaf 开启ceche

    4.nginx动静分离

    5.优化业务逻辑

    6.通过 JVM 参数(如 -Xmx 和 -Xms)进行配置堆区

    nginx动静分离

    将js,css,img等静态资源存放在nginx中,静态资源由nginx返回。

    目录:

    nginx/html/static/index :css,js,img,json

    请求静态资源:

            gulimall.com/static/index/js/catalogLoader.js

    config:

    upstream gulimall {server 192.168.40.1:88;
    }server {listen       80;server_name  gulimall.com;location /static/ {root   /usr/share/nginx/html;}location / {proxy_set_header Host $host;proxy_pass http://gulimall;}#error_page  404              /404.html;# redirect server error pages to the static page /50x.html#error_page   500 502 503 504  /50x.html;location = /50x.html {root   /usr/share/nginx/html;}}   
    

    优化三级分类获取

    原代码13行和19行多次查询数据库,获取子分类list

        @Overridepublic List<CategoryEntity> getLevel1Category() {return this.list(new QueryWrapper<CategoryEntity>().eq("cat_level",1).eq("show_status",1));}@Overridepublic Map<String, List<Level2CategoryVo>> getLevel2AndLevel3Category() {//一级分类List<CategoryEntity> level1 = this.getLevel1Category();Map<String, List<Level2CategoryVo>> categoryMap = level1.stream().collect(Collectors.toMap(k -> k.getCatId().toString(),v ->{//查询该一级分类下的二级分类List<CategoryEntity> level2List = this.list(new QueryWrapper<CategoryEntity>().eq("parent_cid", v.getCatId()));List<Level2CategoryVo> level2CategoryVos = null;if (level2List != null){level2CategoryVos = level2List.stream().map(level2 -> {//查询 二级分类 下的 三级分类List<CategoryEntity> level3List = this.list(new QueryWrapper<CategoryEntity>().eq("parent_cid", level2.getCatId()));List<Level2CategoryVo.Level3Category> collect = null;if (level3List != null) {collect = level3List.stream().map(level3 -> {Level2CategoryVo.Level3Category level3Category = new Level2CategoryVo.Level3Category(level2.getCatId(), level3.getCatId(), level3.getName());return level3Category;}).collect(Collectors.toList());}Level2CategoryVo level2CategoryVo = new Level2CategoryVo(v.getCatId(), collect, level2.getCatId(), level2.getName());return level2CategoryVo;}).collect(Collectors.toList());}return level2CategoryVos;}));return categoryMap;}

    获取全部分类,通过filter获取某分类的子分类list

        @Overridepublic Map<String, List<Level2CategoryVo>> getLevel2AndLevel3Category() {List<CategoryEntity> selectList = this.baseMapper.selectList(null);//一级分类List<CategoryEntity> level1 = this.getByParentCid(selectList,0L);Map<String, List<Level2CategoryVo>> categoryMap = level1.stream().collect(Collectors.toMap(k -> k.getCatId().toString(),v ->{//查询该一级分类下的二级分类List<CategoryEntity> level2List = getByParentCid(selectList,v.getCatId());List<Level2CategoryVo> level2CategoryVos = null;if (level2List != null){level2CategoryVos = level2List.stream().map(level2 -> {//查询 二级分类 下的 三级分类List<CategoryEntity> level3List = getByParentCid(selectList,level2.getCatId());List<Level2CategoryVo.Level3Category> collect = null;if (level3List != null) {collect = level3List.stream().map(level3 -> {Level2CategoryVo.Level3Category level3Category = new Level2CategoryVo.Level3Category(level2.getCatId(), level3.getCatId(), level3.getName());return level3Category;}).collect(Collectors.toList());}Level2CategoryVo level2CategoryVo = new Level2CategoryVo(v.getCatId(), collect, level2.getCatId(), level2.getName());return level2CategoryVo;}).collect(Collectors.toList());}return level2CategoryVos;}));return categoryMap;}private List<CategoryEntity> getByParentCid(List<CategoryEntity> selectList,Long parentCid) {return selectList.stream().filter(categoryEntity -> {return categoryEntity.getParentCid().equals(parentCid); } ).collect(Collectors.toList());}

    Jvm参数配置堆区

    将堆区固定为1024m,Eden区 512m

    测试

    全链路测试,请求首页

    50线程数,吞吐量:1100/s

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

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

    相关文章

    TCP协议支持全双工原因TCP发送接收数据是生产者消费者模型

    一、TCP支持全双工的原因 TCP协议支持全双工&#xff0c;即使用TCP协议进行通信时&#xff0c;服务端和客户端可以同时进行数据的发送和接收&#xff0c;互不干扰&#xff0c;实现同时双向传输数据。 这是因为使用TCP协议通信时&#xff0c;读写套接字的文件描述符既用来发送…

    观成科技:​加密C2框架Platypus流量分析

    一、工具介绍 Platypus 是一款支持多会话的交互式反向 Shell 管理器。在实际的渗透测试中&#xff0c;为了解决 Netcat/Socat 等工具在文件传输、多会话管理方面的不足,该工具在多会话管理的基础上增加了在渗透测试中能更好发挥作用的功能&#xff08;如&#xff1a;交互式 Sh…

    在 C# 中,is null 和 == null ‌不完全等价‌

    最近遇到了一个看似奇怪的问题&#xff0c;判断一个对象是否为null&#xff0c;我使用了null来判断&#xff0c;结果他是null但是仍然进入了判断。 经过讨论和验证&#xff0c;发现使用is null 可以解决问题&#xff0c;于是查阅了资料。 在 C# 中&#xff0c;is null 和 nul…

    go语言zero框架拉取内部平台开发的sdk报错的修复与实践

    在开发过程中&#xff0c;我们可能会遇到由于认证问题无法拉取私有 SDK 的情况。这种情况常发生在使用 Go 语言以及 Zero 框架时&#xff0c;尤其是在连接到私有平台&#xff0c;如阿里云 Codeup 上托管的 Go SDK。如果你遇到这种错误&#xff0c;通常是因为 Go 没有适当的认证…

    VBA+FreePic2Pdf 找出没有放入PDF组合的单个PDF工艺文件

    设计部门针对某个项目做了一个工艺汇总报告&#xff0c;原先只要几十个工艺文件&#xff0c;组合成一个PDF&#xff0c;但后来要求要多放点PDF进去&#xff0c;但工艺文件都混在一起又不知道哪些是重复的&#xff0c;找上我让我帮忙处理一下&#xff0c;我开始建议让她重新再组…

    Webservice如何调用

    webservice调用方式&#xff1a; &#xff08;1&#xff09;http方式调用 请求头增加Content-type:text/xml 或application/soapxml SOAPAction:方法名 请求body以xml字符串传递&#xff0c;xml格式定义 返回以xml字符串返回&#xff0c;xml某个字段是一个json字符串。 入…

    2025-03-10 吴恩达机器学习1——机器学习概述

    文章目录 1 监督学习1.1 回归1.2 分类 2 无监督学习2.1 聚类2.2 异常检测2.3 降维 3 使用 Jupyter Notebook ​ 1959 年&#xff0c;Arthur Samuel 将机器学习定义如下&#xff1a; ​ Field of study that gives computers the ability to learn without being explicitly pro…

    Python 进程与线程-分布式进程

    目录 分布式进程 小结 分布式进程 在Thread和Process中&#xff0c;应当优选Process&#xff0c;因为Process更稳定&#xff0c;而且&#xff0c;Process可以分布到多台机器上&#xff0c;而Thread最多只能分布到同一台机器的多个CPU上。 Python的multiprocessing模块不但支…

    使用 Excel 实现绩效看板的自动化

    引言 在日常工作中&#xff0c;团队的绩效监控和管理是确保项目顺利进行的重要环节。然而&#xff0c;面临着以下问题&#xff1a; ​数据分散&#xff1a;系统中的数据难以汇总&#xff0c;缺乏一个宏观的团队执行情况视图。​看板缺失&#xff1a;系统本身可能无法提供合适…

    Unity中WolrdSpace下的UI展示在上层

    一、问题描述 Unity 中 Canvas使用World Space布局的UI&#xff0c;想让它不被3d物体遮挡&#xff0c;始终显示在上层。 二、解决方案 使用shader解决 在 UI 的材质中禁用深度测试&#xff08;ZTest&#xff09;&#xff0c;强制 UI 始终渲染在最上层。 Shader "Custo…

    uni-app学习笔记——自定义模板

    一、流程 1.这是一个硬性的流程&#xff0c;只要按照如此程序化就可以实现 二、步骤 1.第一步 2.第二步 3.第三步 4.每一次新建页面&#xff0c;都如第二步一样&#xff1b;可以选择自定义的模版&#xff08;vue3Setup——这是我自己的模版&#xff09;&#xff0c;第二步的…

    网络安全防护总体架构 网络安全防护工作机制

    1 实践内容 1.1 安全防范 为了保障"信息安全金三角"的CIA属性、即机密性、完整性、可用性&#xff0c;信息安全领域提出了一系列安全模型。其中动态可适应网络安全模型基于闭环控制理论&#xff0c;典型的有PDR和P^2DR模型。 1.1.1 PDR模型 信息系统的防御机制能…

    tomato靶场通关攻略

    1.打开tomato靶机和kali虚拟机 2.运用kali扫描靶机ip 3.浏览器访问靶机ip 4.kali dirb扫描敏感文件 5.依次查看这几个文件&#xff0c;发现antibot_image下有文件info.php&#xff0c;发现一个被注释的get型文件包含漏洞&#xff0c;参数为image 6.查看日志文件 http://192.…

    玩转云服务器——阿里云操作系统控制台体验测评

    在云服务器日益普及的背景下&#xff0c;运维人员对操作系统管理工具的要求不断提高。我们需要一款既能直观展示系统状态&#xff0c;又能智能诊断问题&#xff0c;提供专业指导的控制台。阿里云操作系统管理平台正是基于API、SDK、CLI等多种管理方式&#xff0c;致力于提升操作…

    Spring boot3-WebClient远程调用非阻塞、响应式HTTP客户端

    来吧&#xff0c;会用就行具体理论不讨论 1、首先pom.xml引入webflux依赖 <dependency><groupId>org.springframework.boot</groupId><artifactId>spring-boot-starter-webflux</artifactId> </dependency> 别问为什么因为是响应式....…

    Qt/C++音视频开发82-系统音量值获取和设置/音量大小/静音

    一、前言 在音视频开发中&#xff0c;音量的控制分两块&#xff0c;一个是控制播放器本身的音量&#xff0c;绝大部分场景都是需要控制这个&#xff0c;这个不会影响系统音量的设置。还有一种场景是需要控制系统的音量&#xff0c;因为播放器本身的音量是在系统音量的基础上控…

    十三、OSG学习笔记-osgDB文件读写

    上一章节&#xff1a; 十二、OSG学习笔记-Control-CSDN博客https://blog.csdn.net/weixin_36323170/article/details/146082287?spm1001.2014.3001.5501 一、文件读取原理 ReadNodeFile,读取流程&#xff1a; 二、实现一个简单的读取器 仿造ReaderWriterOSG.CPP原码&#x…

    05.基于 TCP 的远程计算器:从协议设计到高并发实现

    &#x1f4d6; 目录 &#x1f4cc; 前言&#x1f50d; 需求分析 &#x1f914; 我们需要解决哪些问题&#xff1f; &#x1f3af; 方案设计 &#x1f4a1; 服务器架构 &#x1f680; 什么是协议&#xff1f;为什么要设计协议&#xff1f; &#x1f4cc; 结构化数据的传输问题 …

    设计模式学习笔记——命令模式

    2025年3月13日&#xff0c;周四下午 相同的保存逻辑在各个组件中重复出现。 且需要修改保存逻辑时&#xff0c;各个组件的保存逻辑都需要进行相应修改。 使用了命令模式把保存逻辑从三个组件中独立出来后&#xff0c;减少了代码冗余。 可以通过“保存命令”来使用保存逻辑&am…

    CNN-BiLSTM、BiLSTM、CNN多变量时间序列光伏功率预测Matlab

    CNN-BiLSTM、BiLSTM、CNN多变量时间序列光伏功率预测Matlab 目录 CNN-BiLSTM、BiLSTM、CNN多变量时间序列光伏功率预测Matlab预测效果基本介绍程序设计参考资料 预测效果 基本介绍 CNN-BiLSTM、BiLSTM、CNN三模型多变量时序光伏功率预测 (Matlab2020b 多输入单输出) 1.程序已…