性能测试-JVM配置测试(gc调试)

本文详细介绍了Java虚拟机(JVM)的参数配置,特别是Java_opts的使用方法,涉及堆内存、栈内存大小设置,以及不同垃圾收集器如ParNew、CMS和G1的选择与优化。此外,还探讨了如何通过实例分析解决OutOfMemoryError问题,以及如何通过JMeter进行性能测试和Tomcat配置调优。
摘要由CSDN通过智能技术生成

一、JVM参数配置

Java_opts格式

Java_opts是Java虚拟机(JVM)的一个重要参数,它用于设置JVM的运行参数,例如堆内存大小、栈内存大小、垃圾回收器类型等。在本文中,我们将对Java_opts格式进行科普,并通过示例代码演示其使用方法。

Java_opts格式

Java_opts是一个环境变量,用于设置JVM的运行参数。它的格式为:

java_opts = "-D<property>=<value> -D<property>=<value> ..."

其中,是JVM的属性名称,是对应属性的值。可以使用多个-D=来设置多个属性。
示例

import java.util.Arrays;

public class Main {
    public static void main(String[] args) {
        // 获取Java_opts环境变量
        String javaOpts = System.getenv("JAVA_OPTS");
        
        // 将Java_opts拆分为多个属性
        String[] properties = javaOpts.split(" ");
        
        // 打印每个属性的名称和值
        for (String property : properties) {
            String[] keyValue = property.split("=");
            String name = keyValue[0].substring(2);
            String value = keyValue[1];
            System.out.println("Property: " + name);
            System.out.println("Value: " + value);
        }
    }
}

在这个示例中,我们首先通过System.getenv(“JAVA_OPTS”)获取Java_opts环境变量的值。然后,我们将Java_opts的值拆分为多个属性,使用空格作为分隔符。接下来,我们通过遍历属性数组,将每个属性的名称和值打印出来。

Java_opts的应用

Java_opts可以用于设置各种JVM的运行参数,下面是一些常见的用法:

设置堆内存大小

可以使用-Xmx<value>参数设置JVM的最大堆内存大小,例如-Xmx2g表示最大堆内存为2GB。

设置栈内存大小

tomcat
配置文件:

bin文件夹下: catalina.sh

  • 配置堆栈信息

conf 文件夹下 : server.xml文件

  • 服务、线程池信息

可以使用-Xss参数设置JVM的线程栈内存大小,例如-Xss256k表示栈内存为256KB。
设置垃圾回收器类型
可以使用-XX:+Use参数设置JVM的垃圾回收器类型,例如-XX:+UseG1GC表示使用G1垃圾回收器。
在tomcat文件bin目录下catalina.sh \catalina.bat添加JAVA_OPTS=”-server -Xms512m -Xmx512m -Xmn128m -Xss256k -XX:PermSize=128m -XX:+PrintGC -XX:+PrintGCDetails -XX:+PrintGCTimeStamps -Xloggc:filename”的值修改

cygwin=false
darwin=false
os400=false
hpux=false
case "`uname`" in
CYGWIN*) cygwin=true;;
Darwin*) darwin=true;;
OS400*) os400=true;;
HP-UX*) hpux=true;;
esac
JAVA_OPTS=”-server -Xms512m -Xmx512m -Xmn128m -Xss256k -XX:PermSize=128m -XX:+PrintGC -XX:+PrintGCDetails -XX:+PrintGCTimeStamps -Xloggc:filename”

配置线程池信息
默认:最大线程数量maxThreads=“150” minSpareThreads=“4”
server.xml中线程池的(保持的连接数,企业配置的数值都比较小)
配置 acceptCount: 能接收的队列长度,队列满了再有连接就会被拒绝(修改线程池)

  <Service name="Catalina">

    <!--The connectors can use a shared executor, you can define one or more named thread pools-->
    <!--
    <Executor name="tomcatThreadPool" namePrefix="catalina-exec-"
        maxThreads="150" minSpareThreads="4"/>
    -->

设置系统属性

可以使用-D=参数设置JVM的系统属性,例如-Dfile.encoding=utf-8表示设置文件编码为UTF-8。

总结

ava_opts是Java虚拟机的一个重要参数,用于设置JVM的运行参数。它的格式为-D= -D= …,可以用于设置堆内存大小、栈内存大小、垃圾回收器类型等。在本文中,我们通过示例代码演示了Java_opts的使用方法,并介绍了一些常见的用法。

1、常见参数配置

-XX: PrintGC 每次触发GC的时候打印相关日志
-XX: UseSerialGC 串行回收
-XX: PrintGCDetails 更详细的GC日志
-Xms 堆初始值
-Xmx 堆最大可用值
-Xmn 新生代堆最大可用值
-XX:SurvivorRatio 用来设置新生代中eden空间和from/to空间的比例.
-XX:NewRatio 配置新生代与老年代占比 1:2
含以-XX:SurvivorRatio=eden/from=den/to
总结:在实际工作中,我们可以直接将初始的堆大小与最大堆大小相等,这样的好处是可以减少程序运行时垃圾回收次数,从而提高效率。

-XX:SurvivorRatio 用来设置新生代中eden空间和from/to空间的比例.

2、堆内存大小配置

使用示例: -Xmx20m -Xms5m
说明: 当下Java应用最大可用内存为20M, 初始内存为5M

byte[] b = new byte[4 * 1024 * 1024];
System.out.println("分配了4M空间给数组");
System.out.print("最大内存");
System.out.println(Runtime.getRuntime().maxMemory() / 1024.0 / 1024   "M");
System.out.print("可用内存");
System.out.println(Runtime.getRuntime().freeMemory() / 1024.0 / 1024   "M");
System.out.print("已经使用内存");
System.out.println(Runtime.getRuntime().totalMemory() / 1024.0 / 1024   "M");

3、设置新生代比例参数

在这里插入图片描述

使用示例:-Xms20m -Xmx20m -Xmn1m -XX:SurvivorRatio=2 -XX: PrintGCDetails -XX: UseSerialGC

说明:堆内存初始化值20m,堆内存最大值20m,新生代最大值可用1m,eden空间和from/to空间的比例为2/1

byte[] b = null;
for (int i = 0; i < 10; i  ) {
b = new byte[1 * 1024 * 1024];
}

4、设置新生代与老年代比例参数

使用示例: -Xms20m -Xmx20m -XX:SurvivorRatio=2 -XX: PrintGCDetails -XX: UseSerialGC-XX:NewRatio=2

说明:堆内存初始化值20m,堆内存最大值20m,新生代最大值可用1m,eden空间和from/to空间的比例为2/1新生代和老年代的占比为1/2

二、实战OutOfMemoryError异常

1、Java堆溢出

错误原因: java.lang.OutOfMemoryError: Java heap space 堆内存溢出

解决办法:设置堆内存大小 // -Xms1m -Xmx10m -XX: PrintGCDetails -XX: HeapDumpOnOutOfMemoryError

// -Xms1m -Xmx10m -XX: PrintGCDetails -XX: HeapDumpOnOutOfMemoryError
List<Object> listObject = new ArrayList<>();
for (int i = 0; i < 10; i  ) {
System.out.println("i:"   i);
Byte[] bytes = new Byte[1 * 1024 * 1024];
listObject.add(bytes);
}
System.out.println("添加成功...");

2、虚拟机栈溢出

错误原因: java.lang.StackOverflowError 栈内存溢出

栈溢出 产生于递归调用,循环遍历是不会的,但是循环方法里面产生递归调用, 也会发生栈溢出。

解决办法:设置线程最大调用深度

-Xss5m 设置最大调用深度

public class JvmDemo04 {
     private static int count;
     public static void count(){
        try {
             count  ;
             count(); 
        } catch (Throwable e) {
            System.out.println("最大深度:" count);
            e.printStackTrace();
        }
     }
     public static void main(String[] args) {
         count();
    }
}

3、内存溢出与内存泄漏区别

Java内存泄漏就是没有及时清理内存垃圾,导致系统无法再给你提供内存资源(内存资源耗尽);

而Java内存溢出就是你要求分配的内存超出了系统能给你的,系统不能满足需求,于是产生溢出。

内存溢出,这个好理解,说明存储空间不够大。就像倒水倒多了,从杯子上面溢出了来了一样。

内存泄漏,原理是,使用过的内存空间没有被及时释放,长时间占用内存,最终导致内存空间不足,而出现内存溢出。

三、垃圾收集器

1、串行与并行收集器

串行回收: JDK1.5前的默认算法 缺点是只有一个线程,执行垃圾回收时程序停止的时间比较长

并行回收: 多个线程执行垃圾回收适合于吞吐量的系统,回收时系统会停止运行

2、serial收集器

串行收集器是最古老,最稳定以及效率高的收集器,可能会产生较长的停顿,只使用一个线程去回收。新生代、老年代使用串行回收;新生代复制算法、老年代标记-压缩;垃圾收集的过程中会Stop The World(服务暂停)

一个单线程的收集器,在进行垃圾收集时候,必须暂停其他所有的工作线程直到它收集结束。

特点:CPU利用率最高,停顿时间即用户等待时间比较长。

适用场景:小型应用

通过JVM参数-XX: UseSerialGC可以使用串行垃圾回收器。

3、ParNew收集器

ParNew收集器其实就是Serial收集器的多线程版本。新生代并行,老年代串行;新生代复制算法、老年代标记-压缩

参数控制:-XX: UseParNewGC ParNew收集器

-XX:ParallelGCThreads 限制线程数量

4、parallel 收集器

Parallel Scavenge收集器类似ParNew收集器,Parallel收集器更关注系统的吞吐量。可以通过参数来打开自适应调节策略,虚拟机会根据当前系统的运行情况收集性能监控信息,动态调整这些参数以提供最合适的停顿时间或最大的吞吐量;也可以通过参数控制GC的时间不大于多少毫秒或者比例;新生代复制算法、老年代标记-压缩

采用多线程来通过扫描并压缩堆

特点:停顿时间短,回收效率高,对吞吐量要求高。

适用场景:大型应用,科学计算,大规模数据采集等。

通过JVM参数 XX: USeParNewGC 打开并发标记扫描垃圾回收器。

5、cms收集器

CMS(Concurrent Mark Sweep)收集器是一种以获取最短回收停顿时间为目标的收集器。目前很大一部分的Java应用都集中在互联网站或B/S系统的服务端上,这类应用尤其重视服务的响应速度,希望系统停顿时间最短,以给用户带来较好的体验。

从名字(包含“Mark Sweep”)上就可以看出CMS收集器是基于“标记-清除”算法实现的,它的运作过程相对于前面几种收集器来说要更复杂一些,整个过程分为4个步骤,包括:

初始标记(CMS initial mark)
并发标记(CMS concurrent mark)
重新标记(CMS remark)
并发清除(CMS concurrent sweep)
其中初始标记、重新标记这两个步骤仍然需要“Stop The World”。初始标记仅仅只是标记一下GC Roots能直接关联到的对象,速度很快,并发标记阶段就是进行GC Roots Tracing的过程,而重新标记阶段则是为了修正并发标记期间,因用户程序继续运作而导致标记产生变动的那一部分对象的标记记录,这个阶段的停顿时间一般会比初始标记阶段稍长一些,但远比并发标记的时间短。

由于整个过程中耗时最长的并发标记和并发清除过程中,收集器线程都可以与用户线程一起工作,所以总体上来说,CMS收集器的内存回收过程是与用户线程一起并发地执行。老年代收集器(新生代使用ParNew)

优点:并发收集、低停顿

缺点:产生大量空间碎片、并发阶段会降低吞吐量

采用“标记-清除”算法实现,使用多线程的算法去扫描堆,对发现未使用的对象进行回收。

(1)初始标记
(2)并发标记
(3)并发预处理
(4)重新标记
(5)并发清除
(6)并发重置
特点:响应时间优先,减少垃圾收集停顿时间
适应场景:大型服务器等。

通过JVM参数 -XX: UseConcMarkSweepGC设置

6、g1收集器

在G1中,堆被划分成 许多个连续的区域(region)。采用G1算法进行回收,吸收了CMS收集器特点。

特点:支持很大的堆,高吞吐量

–支持多CPU和垃圾回收线程
–在主线程暂停的情况下,使用并行收集
–在主线程运行的情况下,使用并发收集
实时目标:可配置在N毫秒内最多只占用M毫秒的时间进行垃圾回收

通过JVM参数 -XX: UseG1GC 使用G1垃圾回收器

注意: 并发是指一个处理器同时处理多个任务。

并行是指多个处理器或者是多核的处理器同时处理多个不同的任务。

并发是逻辑上的同时发生(simultaneous),而并行是物理上的同时发生。

来个比喻:并发是一个人同时吃三个馒头,而并行是三个人同时吃三个馒头。

四、Tomcat配置调优测试

1、Jmeter压力测试工具

Meter是一款在国外非常流行和受欢迎的开源性能测试工具,像LoadRunner 一样,它也提供了一个利用本地Proxy Server(代理服务器)来录制生成测试脚本的功能,但是这个功能并不好用。所以在本文中介绍一个更为常用的方法——使用Badboy录制生成 JMeter 脚本。

简单的介绍一下Badboy。Badboy是一款不错的Web自动化测试工具,如果你将它用于非商业用途,或者用于商业用途但是安装Badboy 的机器数量不超过5台,你是不需要为它支付任何费用的。也许是一种推广策略,Badboy提供了将Web测试脚本直接导出生成JMeter 脚本的功能,并且这个功能非常好用,也非常简单。你可以跟着下面的试验步骤来迈出你在开源世界的第一步。

通过Badboy的官方网站下载Badboy的最新版本;
安装Badboy。安装过程同一般的Windows 应用程序没有什么区别,安装完成后你可以在桌面和Windows开始菜单中看到相应的快捷方式——如果找不到,可以找一下Badboy安装目录下的Badboy.exe 文件,直接双击启动Badboy;
启动Badboy,你可以看到下面的界面。
在这里插入图片描述
在地址栏(图中红色方框标注的部分)中输入你需要录制的Web应用的URL——这里我们以http://www.yahoo.com 为例,并点击GO 按钮开始录制。
开始录制后,你可以直接在Badboy内嵌的浏览器(主界面的右侧)中对被测应用进行操作,所有的操作都会被记录在主界面左侧的编辑窗口中——在这个试验中,我们在Yahoo的搜索引擎中输入 JMeter 进行搜索。不过你将看到,录制下来的脚本并不是一行行的代码,而是一个个Web对象——这就有点像LoadRunner的VuGen中的Tree View视图;
录制完成后,点击工具栏中的“停止录制”按钮,完成脚本的录制;
选择“File -> Export to JMeter”菜单,填写文件名“login_mantis.jmx”,将录制好脚本导出为JMeter脚本格式。也可以选择“File -> Save”菜单保存为Badboy脚本;
启动JMeter并打开刚刚生成的测试脚本。

2、什么是吞吐量

QPS:Queries Per Second意思是“每秒查询率”,是一台服务器每秒能够相应的查询次数,是对一个特定的查询服务器在规定时间内所处理流量多少的衡量标准。

3、测试

3.1、测试串行吞吐量

-XX: PrintGCDetails -Xmx32M -Xms1M
-XX: HeapDumpOnOutOfMemoryError
-XX: UseSerialGC
-XX:PermSize=32M
GC 回收次数25次 吞吐量4662 

--> 堆的初始值和堆的最大一致
加大初始堆内存大小-Xms1M 修改为32m
GC 回收次数7次 吞吐量5144

3.2、扩大堆的内存

-XX: PrintGCDetails -Xmx512M -Xms32M
-XX: HeapDumpOnOutOfMemoryError
-XX: UseSerialGC
-XX:PermSize=32M
GC 回收次数6次 吞吐量5141
结论:垃圾回收次数和设置最大堆内存大小无关,只和初始内存有关系。
初始内存会影响吞吐量。

3.3、调整初始堆

-XX: PrintGCDetails -Xmx512M –Xms512M
-XX: HeapDumpOnOutOfMemoryError
-XX: UseSerialGC
-XX:PermSize=32M
GC回收次数0次 吞吐量6561次
结论:堆的初始值和最大堆内存一致,并且初始堆越大就会高。

3.4、并行回收(UseParNewGC)

-XX: PrintGCDetails -Xmx512M -Xms512M
-XX: HeapDumpOnOutOfMemoryError
-XX: UseParNewGC
-XX:PermSize=32M
GC回收0次 吞吐量6800

3.5、CMS收集器

-XX: PrintGCDetails -Xmx512M -Xms512M
-XX: HeapDumpOnOutOfMemoryError
-XX: UseConcMarkSweepGC 
-XX:PermSize=32M

3.6、G1回收方式

-XX: PrintGCDetails -Xmx512M -Xms512M
-XX: HeapDumpOnOutOfMemoryError
-XX: UseG1GC
-XX:PermSize=32M

3、调优总结

初始堆值和最大堆内存内存越大,吞吐量就越高。

最好使用并行收集器,因为并行收集器速度比串行吞吐量高,速度快。

设置堆内存新生代的比例和老年代的比例最好为1:2或者1:3。

减少GC对老年代的回收。

请注意,JAVA_OPTS 环境变量的使用可能会根据您所使用的操作系统和 Java 版本而有所差异。建议查阅相关文档或参考特定的应用程序框架来了解更多关于 JAVA_OPTS 的详细信息和最佳实践。

相关资料:内存知识

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值