实战:使用JMeter的Java Sampler轻松定制自己的benchmark

以前碰到更多的是WEB APP的性能测试,也许用得最多的是Loadrunner, Web Stress tool之类的常态化工具,从没考虑过对于后端引擎类的测试是怎样的。

 

现在有幸主导了一个企业引擎的研发,其最关健的不是其功能如何,而是性能如何。故benchmark就显得尤为重要。

 

既然是benchmark,一系列的性能监测、度量工具使用在所难免:

 1. nmon。首选的AIX/Linux性能监测工具。很多年前简单总结过,可参考:http://www.blogjava.net/itstarting/archive/2009/02/19/255638.html.

 2. jconsole。这个jdk自带的工具,监控JVM Heap size及回收、线程数量、CPU使用之类的,简单好用

 3. JMeter。这个Apache的老字号,不得不搬出来(公司没有Loadrunner,也不能盗版)

 4. Jamon。这个是方法级性能监控工具,完全无侵入性,如果框架用Spring,加上一小段AOP的配置即可

 

 

没想到的是,最给我带来惊喜,还是JMeter.

 

严格意义上上,JMeter我还是第一次在正式场合搬出来深度使用,以前简单的加压http,顶多算是个hello world.

 

JMeter有很多的插件式的扩展,因为本引擎使用的Netty + Protobuf,所以自然想到的还是JUnit Sampler和Java Sample。其中JUnit Sample个人觉得没有Java Sampler好,姑且按下不表。

 

Java Sampler,顾名思义,就是一个Java的采样器,扩展AbstractJavaSamplerClient即可轻松实现。

 

public class Benchmark extends AbstractJavaSamplerClient {
...
}


Benchmark有很多的场景,可以写很多个Benchmark类,比如Benchmark_A, Benchmark_B,我偷懒,四大场景我就写了一个。

 

好了,下面重点说说需要实现的部分,及其作用。

 

1. 重载getDefaultParameters方法。

@Override
public Arguments getDefaultParameters(){
	    Arguments params = new Arguments();
	    
	    //TODO: Right here to define/set parameters from GUI.
	    
	    return params;
}

 在这个方法里,可以设置各种参数,这些参数可以从GUI处获取,也可以设置合适的默认值。有意思的是,这些默认值在GUI扫描的时候,会自动出现在界面上,以供必要的调整和优化。

  

2. 重载setupTest方法。

@Override
public void setupTest(JavaSamplerContext arg0) {
   //TODO: to init the test case, something like JUnit's setUp
}

为了达到真正的性能测试目的,所有的数据准备工作,应在这里完成。 需要注意的是,这个方法在JMeter初始化的每个线程都会执行一遍,而不是全局仅一遍。

  

3. 重载并重点编写runTest方法。

@Override
public SampleResult runTest(JavaSamplerContext arg0) {
    //TODO: The load testing client side logic acting as the emulators puts here
}

这是我们重点要做编写的客户端代码,为Java Sampler实现的重点。

在这里,你可以尽情的调用客户端的API,完成所需要与服务器端交互的一切,包括交易成功与否的判断。

所谓代码胜千言,下面看看代码好了,加上了注释,就不用废话了:

@Override
public SampleResult runTest(JavaSamplerContext arg0) {	
	SampleResult sp = new SampleResult(); //采样结果
        MatchResponse matchResponse;
        UUID uuid = UUID.randomUUID();
        String key = uuid.toString();
        sp.sampleStart();                                    //采用开始时间
        
        //下面是客户端逻辑
        try {        	
        	if("PC2-ID".equalsIgnoreCase(benchmark_mode)){
        		matchResponse = client.sendMatchRequest(
        					key,
							ENGINE.code(), 
							pc2Image, 
							sourcePositions, 
							targetIdentifiers, 
							ImageType.PC2.imageCode());
        	}else if("WSQ-ID".equalsIgnoreCase(benchmark_mode)){
        		matchResponse = client.sendMatchRequest(
    						key,
							ENGINE.code(), 
							wsqImage, 
							sourcePositions, 
							targetIdentifiers, 
							ImageType.WSQ.imageCode());
        	}else if("ID-ID".equalsIgnoreCase(benchmark_mode)){
        		matchResponse = client.sendMatchRequest(
    						key,
							ENGINE.code(), 
							sourceIdentifier, 
							sourcePositions, 
							targetIdentifiers, 
							ImageType.PC2.imageCode());
        	}else{
        		print("Not supported benchmark mode found", benchmark_mode);
        		throw new java.lang.IllegalArgumentException("Not supported benchmark mode found");
        	}

            sp.sampleEnd(); //采用结束
            
            this.printMatchResponse(matchResponse);

        	//set success flag to true but need to verify later
        	sp.setSuccessful(true);
        	
        	//下面进一步分析服务器端返回的信息,以判断本次交易是否成功
        	Iterator<MatchResult> matchResults = matchResponse.getMatchResults().iterator();
        	MatchResult matchResult;
        	MatchScore score;
        	while(matchResults.hasNext()){
        		matchResult = matchResults.next();
        		Iterator<MatchScore> scores = matchResult.getMatchScores().iterator();
        		while(scores.hasNext()){
            		score = scores.next();
            		if (score.score<9999.0){
            			sp.setSuccessful(false);
            			break;
            		}
            	}
        	}
        } catch (Exception e) {
            sp.sampleEnd();
            sp.setSuccessful(false);   
        	e.printStackTrace();
            return sp;   
        }
        
        return sp;  //返回采用结果,这个结果将会被JMeter使用,并反馈到GUI/报告中,方便明了
}


 

4. 重载并编写teardownTest方法。

@Override
public void teardownTest(JavaSamplerContext context) {   
	//TODO: clean up    
}
天下没有不散的筵席,落幕时分跟JUnit的tearDown一样,但据我测试的效果来看,这玩意全局只调用一次,而非跟 setupTest那样每个线程都来一次。

 

 

下面谈谈怎么配置JMeter,并真正跑起来。

 

有几个步骤:

1. 安装JMeter...咳废话

2. 打包Benchmark为jar并拷贝所有的依赖到 %apache-jmeter%\lib\ext下。

我用maven,就更简单了,配上下面那段:

			<plugin>
				<groupId>org.apache.maven.plugins</groupId>
				<artifactId>maven-jar-plugin</artifactId>
				<version>2.3.1</version>
				<configuration>
					<archive>
						<manifest>
							<addClasspath>true</addClasspath>
							<classpathPrefix>lib/</classpathPrefix>
						</manifest>
						<manifestEntries>
				        	<Class-Path>.</Class-Path>
				        </manifestEntries> 
					</archive>
				</configuration>
			</plugin>

这样拷贝过去的就是:

%apache-jmeter%\lib\ext\我的benchmark.jar

%apache-jmeter%\lib\ext\lib\*——各种杂七杂八的依赖

 

3. 启动JMeter。

双击%apache-jmeter%\bin\jmeter.bat

 

4. 新增一个Thread Group.

右键Test Plan->Add->Threads (Users)->Thread Group

改名为Benchmark,并设置几个重量级的参数:

a) Number of Threads (users),并发的线程数
b) Ramp-Up Period (in seconds),预热时间,类似于一般意义上的Think Time
c) Loop Count,每组并发线程循环的次数

d) Scheduler是另外一组用来配置调度的,我没用这个

附上图:


 

5. 新增Java Request.

右键刚新增的线程组(这里名叫Benchmark了)-> Add -> Sampler -> Java Request

默认情况下,如果你仅放了一个Java Sampler,JMeter即可通过扫他的classpath扫到你的Java Sampler并自动出现在GUI上。如下图:


 

6. 再放一到若干个Listener,用以观察输出/图形化报表

一般而言,我喜欢放两个,一个是最最朴素的"View Results in Table":

右键刚新增的线程组Benchmark-> Add -> Listener -> View Results in Table

再来一个"Graph Full Results",图形化的

 

到此为止,基本配置完毕,保存一下。

 

 

跑Benchmark其实就很简单了,一般而言三个步骤:

1. 调整Thread Group的参数,一般包括Number of Threads (users)和Loop Count,以配合不同的压力场景

2. 调整必要的参数,通过点击Java Request,即可控制你的Java Sampler的每个参数细节

3. 点击Start,并观察输出,包括View Results in Table或者Graph Full Results。输出主要包括:No of Samples、Latest Sample、Average、Deviation、Throughput、Median等,够用了。如下图:


 

 

附:一般而言,如果线程不是太吓人(比如超过1000),一个JMeter实例就够了,如果多了,就要考虑多个JMeter实例并行测试了。

 

==The End==

  • 0
    点赞
  • 3
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
### 回答1: JMeter BeanShell Sampler是一种JMeterSampler,它使用BeanShell脚本语言编写。BeanShell是一种Java编写的脚本语言,它可以在JVM上运行,并且可以访问Java类和接口。使用BeanShell脚本,可以在JMeter中编写自定义的Sampler,以便执行更复杂的测试任务。BeanShell Sampler可以通过JMeter的GUI界面或者JMeter的脚本文件进行配置和使用。 ### 回答2: JMeter是一个性能测试工具,可以模拟大量用户对Web应用程序的访问,测试系统的性能。Beanshell Sampler则是JMeter的一个非常有用的组件,它允许测试人员使用Beanshell脚本来执行复杂的测试逻辑或对测试数据进行处理。 使用Beanshell Sampler的第一步是将它添加到JMeter测试计划中。可以在JMeter工具栏中选择“Sampler”菜单,然后选择“Beanshell Sampler”。在Beanshell Sampler配置页面中,可以编写自定义的Beanshell脚本。脚本可以使用Java语法编写,可以使用Beanshell语法,也可以使用Java类库。 一个简单的Beanshell脚本可以像这样: ``` log.info("Hello, World!"); ``` 这个简单的脚本会将“Hello, World!”信息写入到JMeter的日志文件中。 Beanshell脚本既可以在每个请求之前执行,也可以在每个请求之后执行。默认情况下,Beanshell Sampler在每个请求之前执行。如果需要在每个请求之后执行,则需要将“Run thread group as a loop”选项设置为False,并将“Action to be taken after a sampler error”选项设置为“Continue”. 在脚本中,可以使用很多JMeter提供的对象和方法,如vars, log, prev,ctx等。vars对象允许在脚本中使用JMeter变量,log对象可以记录日志信息。 ``` log.info("Response code: " + prev.getResponseCode()); log.info("Response message: " + prev.getResponseMessage()); ``` prev对象包含了上一个请求的响应数据,ctx对象包含了JMeter上下文信息。 除了使用JMeter内置对象和方法,Beanshell还可以使用Java类库的方法,只需要将需要使用Java类库的jar文件放置在JMeter的lib目录下即可。 总之,Beanshell Sampler是JMeter非常有用的组件,通过编写Beanshell脚本,可以实现很多复杂的测试逻辑和数据处理操作,从而更好地完成性能测试任务。 ### 回答3: JMeter Beanshell Sampler是JMeter的一个组件,可以用于执行任意Java代码。使用JMeter Beanshell Sampler可以方便地对JMeter进行扩展和自定义。在JMeter中,可以通过以下步骤使用Beanshell Sampler: 1. 新建一个测试计划,然后选择添加“Thread Group”组件。 2. 在“Thread Group”组件中添加“Beanshell Sampler”组件。 3. 在“Beanshell Sampler”组件中编写Java代码。 4. 运行测试计划,查看测试结果。 在编写Beanshell脚本时,可以使用JMeter提供的Beanshell变量和方法,也可以使用自定义变量和方法。此外,还可以在Beanshell脚本中调用外部的Java类和方法。需要注意的是,Beanshell脚本的性能要比纯Java代码稍差,因为Beanshell是解释执行的。 在使用Beanshell Sampler时,可以实现一些常见的场景,例如: 1. 计算某些数据后再发送HTTP请求。 2. 根据条件控制请求的发送次数。 3. 根据特定的逻辑生成请求数据。 4. 在JMeter中实现自定义函数等功能。 需要注意的是,在使用Beanshell Sampler时需要谨慎操作,避免因为编写不当而导致性能问题或安全问题。因此,编写Beanshell脚本时需要遵循Java的最佳实践,尽量减少不必要的计算和I/O操作,同时进行充分的测试和验证。

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

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

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值