淺談Spring Boot: 接口壓測(cè)及簡(jiǎn)要優(yōu)化策略
工程做好之后,需要對(duì)接口進(jìn)行壓力測(cè)試。可以自己編寫線程池模擬多用戶訪問(wèn)測(cè)試,也可以使用jmeter進(jìn)行壓測(cè)。jmeter的好處是測(cè)試方便,并且有完善的結(jié)果分析功能。本次采用jmeter進(jìn)行壓力測(cè)試。
1.準(zhǔn)備數(shù)據(jù),為了測(cè)試準(zhǔn)備200w條以上的數(shù)據(jù)。一個(gè)簡(jiǎn)單的方法是使用下面的sql快速創(chuàng)建。
INSERT INTO table (user_name,address)
SELECT user_name, address FROM table;
但這樣創(chuàng)建的數(shù)據(jù)不同記錄的重復(fù)部分太多,和實(shí)際業(yè)務(wù)不太相符。一般業(yè)務(wù)上,除了主鍵之外還會(huì)有某一個(gè)字段是唯一,比如手機(jī)號(hào),用戶名等。本次將user_name設(shè)置為唯一,簡(jiǎn)單采用UUID的方式生成。
@RequestMapping('/create')public Integer createData(Integer password) {if (password != 1024) {return 0;}ThreadPoolExecutor poolExecutor = new ThreadPoolExecutor(10,20, 1, TimeUnit.MINUTES,new ArrayBlockingQueue<Runnable>(100000),Executors.defaultThreadFactory(), new ThreadPoolExecutor.CallerRunsPolicy());int max = 1000000;for (int i = 0; i < max; i++) {poolExecutor.execute(() -> {Date now = new Date();User user = new User();user.setStatus(0);user.setUserName(UUID.randomUUID().toString().replace('-', ''));user.setAddress('');user.setCreateTime(now);user.setUpdateTime(now);userService.saveUser(user);});}return 1;}
采用線程池技術(shù)來(lái)生成數(shù)據(jù)。部分參數(shù)參考自己的配置進(jìn)行設(shè)置。我這里采用核心線程數(shù)10,最大線程數(shù)20,阻塞隊(duì)列容量10w,拒絕策略CallerRunsPolicy的參數(shù)來(lái)生成。
數(shù)據(jù)生成以后,確認(rèn)下生成數(shù)量,可以看到所有的user_name都是不同的。
select count(*),count(distinct user_name) from user

2.改造接口。為了測(cè)試方便,使用公共mapper。引入依賴
<dependency><groupId>tk.mybatis</groupId><artifactId>mapper-spring-boot-starter</artifactId></dependency>
修改啟動(dòng)類MapperScan注解為import tk.mybatis.spring.annotation.MapperScan;
mapper接口繼承extends Mapper,(import tk.mybatis.mapper.common.Mapper;)。這樣就會(huì)為mapper生成insert,select等基本方法。
3.測(cè)試。
所測(cè)試的接口為
@RequestMapping(value = '/hello', method = {RequestMethod.POST})public List<User> getUser(User user) {return userService.getUserByUser(user);}
打開(kāi)jmeter,通過(guò)Option選擇中文語(yǔ)言。創(chuàng)建測(cè)試計(jì)劃,這次僅對(duì)接口進(jìn)行壓力測(cè)試。
測(cè)試參數(shù)選擇userName,并使用隨機(jī)生成的UUID,這樣可以保證最終訪問(wèn)DB時(shí)不觸發(fā)任何緩存。
1.文件-新建測(cè)試計(jì)劃
2.編輯-添加-線程-線程組
3.選中線程組,編輯-添加-邏輯控制器-事務(wù)控制器
4.選中事務(wù)控制器,編輯-添加-取樣器-BeanShell 取樣器
5.選中事務(wù)控制器,編輯-添加-HTTP請(qǐng)求
6.選中BeanShell取樣器,將下面代碼復(fù)制到腳本框里面,內(nèi)容即為設(shè)置user_name變量為UUID。
import java.util.UUID;UUID uuid1 = UUID.randomUUID();vars.put('user_name',(uuid1.toString()).toUpperCase().replaceAll('-',''));

7.選中HTTP請(qǐng)求,設(shè)置如下,其中參數(shù)部分內(nèi)容為:
名稱 值 userName ${user_Name}這樣就可以將步驟6生成的參數(shù)傳遞為HTTP POST請(qǐng)求的參數(shù)了。

添加結(jié)果樹,匯總報(bào)告等,最終結(jié)構(gòu):

8.選中線程組,設(shè)置線程數(shù)等信息,詳細(xì)請(qǐng)參照官網(wǎng):https://jmeter.apache.org/usermanual/index.html

這里先設(shè)置線程數(shù)為2000,由于要壓力測(cè)試,設(shè)置Ramp-up為1秒,即1秒內(nèi)啟動(dòng)所有線程。
9.啟動(dòng)線程組。在這里插入圖片描述
10.結(jié)果樹里面可以確認(rèn)具體每次請(qǐng)求的參數(shù):

匯總報(bào)告里面可以查看總體信息:

通過(guò)以上步驟,可以初步預(yù)估下系統(tǒng)各個(gè)接口的吞吐量等信息。第一次壓測(cè)中user_name自動(dòng)沒(méi)有設(shè)置索引,可以通過(guò)一些方法來(lái)提高系統(tǒng)性能:
1.為user_name設(shè)置索引。
2.使用redis緩存,其一是緩存最近檢索數(shù)據(jù),其二是將表中所有user_name存入緩存(Set),請(qǐng)求到來(lái)時(shí)首先去緩存中查看是否存在,只有存在的時(shí)候才去檢索DB。
3.若數(shù)據(jù)量過(guò)大,可采用布隆過(guò)濾器存儲(chǔ)user_name。
4.限流,這里只說(shuō)API層面的,guava包下RateLimiterJ;自寫切面+Redis;Spring Cloud GateWay。
工程地址:https://github.com/showsys20/spring-demo-cm.git
以上這篇淺談Spring Boot: 接口壓測(cè)及簡(jiǎn)要優(yōu)化策略就是小編分享給大家的全部?jī)?nèi)容了,希望能給大家一個(gè)參考,也希望大家多多支持好吧啦網(wǎng)。
相關(guān)文章:
1. IDEA 2020.1.2 安裝教程附破解教程詳解2. Java利用TCP協(xié)議實(shí)現(xiàn)客戶端與服務(wù)器通信(附通信源碼)3. 40個(gè)Java集合面試問(wèn)題和答案4. Spring如何集成ibatis項(xiàng)目并實(shí)現(xiàn)dao層基類封裝5. django queryset相加和篩選教程6. idea設(shè)置提示不區(qū)分大小寫的方法7. IntelliJ IDEA導(dǎo)出項(xiàng)目的方法8. ajax實(shí)現(xiàn)頁(yè)面的局部加載9. 使用AJAX(包含正則表達(dá)式)驗(yàn)證用戶登錄的步驟10. JS圖片懶加載庫(kù)VueLazyLoad詳解

網(wǎng)公網(wǎng)安備