springboot項(xiàng)目組引入JMeter的實(shí)現(xiàn)步驟
一、前言
看到標(biāo)題的小伙伴肯定很好奇,springboot項(xiàng)目中用JMeter能干什么,那么我先反問(wèn)你JMeter能用來(lái)作什么?壓測(cè);接口自動(dòng)化?,如果你能猜到這兩點(diǎn)那你對(duì)JMeter肯定是有了一定的認(rèn)知,同樣你也猜對(duì)了,用它來(lái)做接口自動(dòng)化
但是可能會(huì)有小伙伴會(huì)有疑問(wèn)為什么不拿他來(lái)做壓測(cè),當(dāng)然可能拿來(lái)做壓測(cè)但是不建議在springboot項(xiàng)目中整合壓測(cè)功能,畢竟壓測(cè)本來(lái)就是一種多線程的形式而且極其消耗內(nèi)存,spring項(xiàng)目本來(lái)就很重,所以得不償失。當(dāng)然對(duì)于壓測(cè)我又另一種方案,就是容器化的分布式壓測(cè),這一塊方案我會(huì)再后續(xù)的文章分享出,接下來(lái)我們先來(lái)談?wù)勅绾卧趕pringboot項(xiàng)目中做接口自動(dòng)化吧
在這篇文章中我詳細(xì)介紹過(guò)JMeter engine啟動(dòng)原理,知道JMeter engine原理之后,是否我們可以這樣做,在springboot項(xiàng)目引入JMeter開(kāi)源組件包然后使用其開(kāi)啟engine的方法,自己開(kāi)啟engine,當(dāng)然此時(shí)你的線程肯定是單線程,所以基本內(nèi)存消耗很低,那么如何做?我們接著往下說(shuō)
二、springboot項(xiàng)目引入JMeter完成接口自動(dòng)化操作
首先導(dǎo)入JMeter對(duì)應(yīng)的maven坐標(biāo)
<properties>
<jmeter.version>5.3</jmeter.version>
</properties>
<dependencies>
<dependency>
<groupId>org.apache.jmeter</groupId>
<artifactId>ApacheJMeter_http</artifactId>
<version>${jmeter.version}</version>
<exclusions>
<exclusion>
<groupId>org.apache.logging.log4j</groupId>
<artifactId>log4j-slf4j-impl</artifactId>
</exclusion>
</exclusions>
</dependency>
<dependency>
<groupId>org.apache.jmeter</groupId>
<artifactId>ApacheJMeter_functions</artifactId>
<version>${jmeter.version}</version>
</dependency>
<dependency>
<groupId>org.apache.jmeter</groupId>
<artifactId>ApacheJMeter_jdbc</artifactId>
<version>${jmeter.version}</version>
</dependency>
<dependency>
<groupId>org.apache.jmeter</groupId>
<artifactId>ApacheJMeter_tcp</artifactId>
<version>${jmeter.version}</version>
</dependency>
</dependencies>
在JMeter engine啟動(dòng)原理介紹了StandardJMeterEngine標(biāo)準(zhǔn)的引擎類,通過(guò)創(chuàng)建這個(gè)對(duì)象就可以開(kāi)啟engine,當(dāng)然事實(shí)并非如此簡(jiǎn)單
//創(chuàng)建engine
StandardJMeterEngine engine = new StandardJMeterEngine();
//將hashTree配置注入engine中
engine.configure(hashTree);
try {
//執(zhí)行engine開(kāi)始運(yùn)行
engine.runTest();
} catch (JMeterEngineException e) {
engine.stopTest();
e.printStackTrace();
}
所以開(kāi)啟engine就這么短短的幾步操作就行,但是上面需要HashTree對(duì)象傳入,就是你的配置,關(guān)于HashTree的用法及使用,可以參考我寫的這篇博文

HashTree有了之后,理論上是可以運(yùn)行了,但是實(shí)際卻會(huì)報(bào)錯(cuò),因?yàn)镴Meter engine開(kāi)啟會(huì)初始化配置文件,也就是我們熟知的jmeter.properties配置的內(nèi)容,所以你可以在springboot項(xiàng)目的resource下創(chuàng)建jmeter目錄存放jmeter.properties等配置
配置文件可以去github上去獲取JMeter 配置

好了配置有了,接下來(lái)我們需要先初始化jmeter.properties配置屬性然后在開(kāi)啟jmeter engine就行
//初始化jmeter屬性配置
private void initJMeterProperties() {
if (!StringUtils.isEmpty(JMeterUtils.getJMeterProperties())){
return;
}
try {
InputStream inputStream = JMeterUtil.class.getResource("/jmeter/jmeter.properties").openStream();
File tempFile = FileUtil.createTempFile(null);
FileUtil.writeFromStream(inputStream,tempFile);
//這里面loadJMeterProperties方法必須寫成臨時(shí)文件這樣的形式,否則會(huì)獲取不到j(luò)meter.properties
JMeterUtils.loadJMeterProperties(tempFile.getAbsolutePath());
JMeterUtils.setJMeterHome(JMeterUtil.PATH);
JMeterUtils.setLocale(LocaleContextHolder.getLocale());
} catch (IOException e) {
e.printStackTrace();
}
}
在上面代碼中必須通過(guò)創(chuàng)建臨時(shí)文件的形式傳入臨時(shí)文件的路徑,否則springboot打成jar無(wú)法找到路徑從而報(bào)錯(cuò)

在執(zhí)行步驟2即可,但是會(huì)有一個(gè)問(wèn)題engine.runTest()是一個(gè)空方法,那你如何接收運(yùn)行的結(jié)果集數(shù)據(jù)呢?這里就要牽扯到jmeter后端監(jiān)聽(tīng)器的作用了,JMeter后端監(jiān)聽(tīng)器作用及原理可以參考這篇文章
后端監(jiān)聽(tīng)器可以異步的接收jmeter engine運(yùn)行的結(jié)果集,所以我們可以通過(guò)這個(gè)方式實(shí)現(xiàn)一個(gè)完整的接口自動(dòng)化流程
這是關(guān)于我的springboot項(xiàng)目采用后端監(jiān)聽(tīng)器的實(shí)現(xiàn)方式
public class CustomBackendListenerClient extends AbstractBackendListenerClient {
private final List<SampleResult> queue = new ArrayList<>();
//獲得控制臺(tái)內(nèi)容。
private PrintStream oldPrintStream = System.out;
private ByteArrayOutputStream bos = new ByteArrayOutputStream();
private void setConsole() {
System.setOut(new PrintStream(bos));
}
private String getConsole() {
System.setOut(oldPrintStream);
return bos.toString();
}
// engine執(zhí)行之前會(huì)進(jìn)行前置處理器
@Override
public void setupTest(BackendListenerContext context) throws Exception {
setConsole();
super.setupTest(context);
}
//engine執(zhí)行中的處理
@Override
public void handleSampleResults(List<SampleResult> sampleResults, BackendListenerContext context) {
//結(jié)果集添加至集合中
queue.addAll(sampleResults);
}
//engine結(jié)束后的后置處理器
@Override
public void teardownTest(BackendListenerContext context) throws Exception {
//處理結(jié)果集中的數(shù)據(jù)并封裝至JMeterRequestResult對(duì)象中
RunJMeterRequestService runJMeterRequestService = CommonBeanFactory.getBean(RunJMeterRequestService.class);
List<JMeterRequestResult> jMeterRequestResults = Lists.newArrayList();
String testId = context.getParameter("testId");
queue.stream().forEach(result -> {
setRequestResult(result, jMeterRequestResults);
});
queue.clear();
runJMeterRequestService.addDebugResult(testId, jMeterRequestResults);
super.teardownTest(context);
}
private void setRequestResult(SampleResult result, List<JMeterRequestResult> jMeterRequestResults) {
JMeterRequestResult metricResult = new JMeterRequestResult();
Long responseTime = result.getEndTime() - result.getStartTime();
metricResult.setUrl(result.getUrlAsString());
metricResult.setResponseSize(((Integer) result.getBodySize()).toString());
metricResult.setResponseTime(responseTime.toString());
metricResult.setResponseResult(result.getResponseDataAsString());
metricResult.setConsoleResult(getConsole());
if (result instanceof HTTPSampleResult) {
HTTPSampleResult res = (HTTPSampleResult) result;
metricResult.setCookie(res.getCookies());
metricResult.setRequestMethod(res.getHTTPMethod());
}
metricResult.setRequestData(result.getSamplerData());
metricResult.setResponseHeader(result.getResponseHeaders());
metricResult.setRequestHeader(result.getRequestHeaders());
metricResult.setStatusCode(result.getResponseCode());
jMeterRequestResults.add(metricResult);
}
}
至此JMeter接口自動(dòng)化可以通過(guò)此方案來(lái)實(shí)現(xiàn),希望能對(duì)你有所啟發(fā)
到此這篇關(guān)于springboot項(xiàng)目組引入JMeter的實(shí)現(xiàn)步驟的文章就介紹到這了,更多相關(guān)springboot項(xiàng)目組引入JMeter內(nèi)容請(qǐng)搜索腳本之家以前的文章或繼續(xù)瀏覽下面的相關(guān)文章希望大家以后多多支持腳本之家!
相關(guān)文章
Java實(shí)戰(zhàn)權(quán)限管理系統(tǒng)的實(shí)現(xiàn)流程
讀萬(wàn)卷書不如行萬(wàn)里路,只學(xué)書上的理論是遠(yuǎn)遠(yuǎn)不夠的,只有在實(shí)戰(zhàn)中才能獲得能力的提升,本篇文章手把手帶你用java+SpringBoot+MyBatis+AOP+LayUI+Mysql實(shí)現(xiàn)一個(gè)權(quán)限管理系統(tǒng),大家可以在過(guò)程中查缺補(bǔ)漏,提升水平2022-01-01
JAVA String.valueOf()方法的用法說(shuō)明
這篇文章主要介紹了JAVA String.valueOf()方法的用法說(shuō)明,具有很好的參考價(jià)值,希望對(duì)大家有所幫助。一起跟隨小編過(guò)來(lái)看看吧2020-09-09
Java?多個(gè)文件生成zip包、下載zip包的實(shí)現(xiàn)代碼
這篇文章主要介紹了Java?多個(gè)文件生成zip包、下載zip包,包括文件上傳,文件下載,多個(gè)文件打成zip包的操作代碼,本文給大家介紹的非常詳細(xì),需要的朋友可以參考下2024-01-01
Java實(shí)現(xiàn)Excel轉(zhuǎn)PDF的兩種方法詳解
使用具將Excel轉(zhuǎn)為PDF的方法有很多,在這里我給大家介紹兩種常用的方法:使用spire轉(zhuǎn)化PDF、使用jacob實(shí)現(xiàn)Excel轉(zhuǎn)PDF,分別應(yīng)對(duì)兩種不一樣的使用場(chǎng)景,需要的可以參考一下2022-01-01
tk.mybatis通用插件updateByPrimaryKeySelective無(wú)法自動(dòng)更新列的解決辦法
tk.mybatis是一個(gè)很好用的通用插件,本文主要介紹了tk.mybatis通用插件updateByPrimaryKeySelective無(wú)法自動(dòng)更新列的解決辦法,具有一定的參考價(jià)值,感興趣的小伙伴們可以參考一下2021-12-12
idea如何在service窗口中顯示多個(gè)服務(wù)
這篇文章主要介紹了idea如何在service窗口中顯示多個(gè)服務(wù)問(wèn)題,具有很好的參考價(jià)值,希望對(duì)大家有所幫助,如有錯(cuò)誤或未考慮完全的地方,望不吝賜教2023-09-09
java開(kāi)發(fā)微服務(wù)架構(gòu)設(shè)計(jì)消息隊(duì)列的水有多深
今天我們說(shuō)說(shuō)消息隊(duì)列的問(wèn)題,來(lái)帶大家探一探消息隊(duì)列的水有多深,希望看完本文大家在引入消息隊(duì)列的時(shí)候先想一想,是不是一定要引入?引入消息隊(duì)列后產(chǎn)生的問(wèn)題能不能解決2021-10-10

