python單鏈路性能測試實(shí)踐
引言
在經(jīng)歷過一些嘗試之后,覺得在當(dāng)下的項(xiàng)目中運(yùn)用鏈路壓測的能力,不等著其他人了。
鏈路這個詞其實(shí)不如路徑通俗易懂,跟產(chǎn)品溝通這個比較有效率。具體的操作路徑,產(chǎn)品會給一份出來,但是這都是基于UI和產(chǎn)品思維的文檔,跟接口測試區(qū)別還是很大的,只能提供參考依據(jù)。
需要端上測試協(xié)作,有些業(yè)務(wù)細(xì)節(jié)還得端上測試同學(xué)幫忙補(bǔ)充一下。還需要運(yùn)維同事幫忙理一下各個接口的請求量比例,這次的比例我是依據(jù)靈光一現(xiàn)寫出來,然后大家一起調(diào)整的。
本次由于比較初級,所以這塊文檔就不寫出來了,放一個圖來表達(dá)一下這個鏈路做了些什么,PS:我現(xiàn)在很喜歡用圖而不是文字,溝通效率太高了。推薦工具draw.io,感興趣的可以參考文末的熱文中兩張架構(gòu)圖中的介紹。

資源庫1.4鏈路壓測方案
這次把登錄剔除了,因?yàn)樘耍瑢y試結(jié)果影響比較大。
場景思路
場景
場景就是老師登錄,首先會請求一個知識點(diǎn)列表,然后通過知識點(diǎn)屬性篩選推薦課程列表,在對課程列表中的數(shù)據(jù)進(jìn)行收藏和取消收藏,在獲取自己當(dāng)前知識點(diǎn)下的課程列表(包含原創(chuàng)和收藏)。
思路
本次依然采取固定線程的壓測模型,本人預(yù)估線程200左右,測試用戶600備用,列表頁保證2頁數(shù)據(jù)。
每個線程綁定一個用戶,然后用戶開始循環(huán)鏈路執(zhí)行步驟,執(zhí)行一次當(dāng)做一次Q。單次Q包含9次HTTP接口請求(放棄了Socket接口,以后有需求再添加Socket接口到鏈路中),其中3次修改操作,6次查詢操作。
具體的邏輯通過內(nèi)部靜態(tài)類實(shí)現(xiàn),然后多一個K類,用來存儲每次獲取的知識點(diǎn)屬性,方便調(diào)用。由于接口請求方法都是用基礎(chǔ)數(shù)據(jù)類型和String作為參數(shù),所以調(diào)用時候會顯得有點(diǎn)啰嗦。但無傷大雅,腳本寫出來,本來就是用完就扔到倉庫里面,改天再用再優(yōu)化。
Demo實(shí)現(xiàn)
package?com.okayqa.composer.performance.resource1_4
import?com.alibaba.fastjson.JSON
import?com.alibaba.fastjson.JSONObject
import?com.funtester.base.bean.AbstractBean
import?com.funtester.base.constaint.ThreadLimitTimesCount
import?com.funtester.frame.execute.Concurrent
import?com.funtester.httpclient.ClientManage
import?com.funtester.utils.ArgsUtil
import?com.okayqa.composer.base.OkayBase
import?com.okayqa.composer.function.Mirro
import?com.okayqa.composer.function.OKClass
class?Login_collect_uncollect?extends?OkayBase?{
????public?static?void?main(String[]?args)?{
????????ClientManage.init(10,?5,?0,?"",?0)
????????def?util?=?new?ArgsUtil(args)
????????def?thread?=?util.getIntOrdefault(0,?30)
????????def?times?=?util.getIntOrdefault(1,?40)
????????def?tasks?=?[]
????????thread.times?{
????????????tasks?<<?new?FunTester(it,?times)
????????}
????????new?Concurrent(tasks,?"資源庫1.4登錄>查詢>收藏>取消收藏鏈路壓測").start()
????????allOver()
????}
????private?static?class?FunTester?extends?ThreadLimitTimesCount<Integer>?{
????????OkayBase?base
????????def?mirro
????????def?clazz
????????FunTester(Integer?integer,?int?times)?{
????????????super(integer,?times,?null)
????????}
????????@Override
????????void?before()?{
????????????super.before()
????????????base?=?getBase(t)
????????????mirro?=?new?Mirro(base)
????????????clazz?=?new?OKClass(base)
????????}
????????@Override
????????protected?void?doing()?throws?Exception?{
????????
????????????def?klist?=?mirro.getKList()</code><code>????????????mirro.getKList()
????????????def?karray?=?klist.getJSONArray("data")
????????????K?ks
????????????karray.each?{
????????????????JSONObject?parse?=?JSON.parse(JSON.toJSONString(it))
????????????????if?(ks?==?null)?{
????????????????????def?level?=?parse.getIntValue("node_level")
????????????????????def?type?=?parse.getIntValue("ktype")
????????????????????def?id?=?parse.getIntValue("id")
????????????????????ks?=?new?K(id,?type,?level)
????????????????}
????????????}
????????????JSONObject?response?=?clazz.recommend(ks.id,?ks.type,?ks.level)</code><code>????????????clazz.recommend(ks.id,?ks.type,?ks.level)????????????clazz.recommend(ks.id,?ks.type,?ks.level)
????????????def?minis?=?[]
????????????int?i?=?0
????????????response.getJSONArray("data").each?{
????????????????if?(i++?<?2)?{
????????????????????JSONObject?parse?=?JSON.parse(JSON.toJSONString(it))
????????????????????int?value?=?parse.getIntValue("minicourse_id")
????????????????????minis?<<?value
????????????????}
????????????}
????????????clazz.unCollect(random(minis))
????????????mirro.getMiniCourseListV3(ks.id,?ks.type,?0,?ks.level)????????????mirro.getMiniCourseListV3(ks.id,?ks.type,?0,?ks.level)
????????}
????}
????private?static?class?K?extends?AbstractBean?{
????????int?id
????????int?type
????????int?level
????????K(int?id,?int?type,?int?level)?{
????????????this.id?=?id
????????????this.type?=?type
????????????this.level?=?level
????????}
????}
}
其中AbstractBean類是一個抽象類,用于一些bean的方法封裝,就是為了省事兒。
package?com.funtester.base.bean
import?com.alibaba.fastjson.JSON
import?com.alibaba.fastjson.JSONObject
import?com.funtester.frame.Save
import?com.funtester.frame.SourceCode
import?org.slf4j.Logger
import?org.slf4j.LoggerFactory
import?org.springframework.beans.BeanUtils
/**
?*?bean的基類
?*/
abstract?class?AbstractBean?{
????static?final?Logger?logger?=?LoggerFactory.getLogger(AbstractBean.class)
????/**
?????*?將bean轉(zhuǎn)化為json,為了進(jìn)行數(shù)據(jù)處理和打印
?????*
?????*?@return
?????*/
????JSONObject?toJson()?{
????????JSONObject.parseObject(JSONObject.toJSONString(this))
????}
????/**
?????*?文本形式保存
?????*/
????def?save()?{
????????Save.saveJson(this.toJson(),?this.getClass().toString()?+?SourceCode.getMark());
????}
????/**
?????*?控制臺打印,使用WARN記錄,以便查看
?????*/
????def?print()?{
????????logger.warn(this.getClass().toString()?+?":"?+?this.toString());
????}
????def?initFrom(String?str)?{
????????JSONObject.parseObject(str,?this.getClass())
????}
????def?initFrom(Object?str)?{
????????initFrom(JSON.toJSONString(str))
????}
????def?copyFrom(AbstractBean?source)?{
????????BeanUtils.copyProperties(source,?this)
????}
????def?copyTo(AbstractBean?target)?{
????????BeanUtils.copyProperties(this,?target)
????}
????/**
?????*?這里bean的屬性必需是可以訪問的,不然會返回空json串
?????*?@return
?????*/
????@Override
????String?toString()?{
????????JSONObject.toJSONString(this)
????}
????@Override
????protected?Object?clone()?{
????????initFrom(this)
????}
}
控制臺輸出
~?~~?~~?~~?~~?~~?~~?~~?~~?~~?~ JSON ~?~~?~~?~~?~~?~~?~~?~~?~~?~~?~
> {
> ① . "rt":1665,
> ① . "total":1188,
> ① . "qps":18.018,
> ① . "failRate":0.0,
> ① . "threads":30,
> ① . "startTime":"2021-02-24 16:57:23",
> ① . "endTime":"2021-02-24 16:58:34",
> ① . "errorRate":1.01,
> ① . "executeTotal":1188,
> ① . "mark":"資源庫1.4登錄>查詢>收藏>取消收藏鏈路壓測241657",
> ① . "table":"eJzj5VLAD15sbXm2a8LTXZMN9Uyez9z9dO9Uu2fzl75Yv8ju2ZRtL6b32z3tn/ZsWweE83Lyvhfb1z/t6362tZvT2EChJKMoNaWYgA0KvFy8+F0RlFpckJ9XnKoQkpmbaqVQoVucWpSZmKOQV5qro1Cpm5uakpmYR8gOQq5QyM3MU4AYZWVhYqmQW6yTm1hhZWxoaQxkE9RNjA2UgEfTOoBo1JZRW2hmRSsQ0ccmsBW0tgnVQzS1DatVtLMRn3W0sPXRtCYgAlLtQITXWura/mhaMxCRYC+VXUGyv2nhmkfTGoGI0rCgrsseTWsBImLSIZ1dCA8sOjmMXNfCU9ZAupNYV8MdC4106qdA2vkAniAGq6OJ8AlVi6GB99FgSvTU8BU8egaBg6jsO3iWHwSuoZUPB4EzRn046sNRHw68M0Z9OOrDEe5DABkr1eo="
> }
~?~~?~~?~~?~~?~~?~~?~~?~~?~~?~ JSON ~?~~?~~?~~?~~?~~?~~?~~?~~?~~?~

以上就是python單鏈路性能測試實(shí)踐的詳細(xì)內(nèi)容,更多關(guān)于python單鏈路性能測試的資料請關(guān)注腳本之家其它相關(guān)文章!
相關(guān)文章
Python使用jpype模塊調(diào)用jar包過程解析
這篇文章主要介紹了Python使用jpype模塊調(diào)用jar包過程解析,文中通過示例代碼介紹的非常詳細(xì),對大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價值,需要的朋友可以參考下2020-07-07
Python中使用json.load()和json.loads()加載json數(shù)據(jù)的方法實(shí)例
在python編程中,我們經(jīng)常要用到j(luò)son對象作為數(shù)據(jù)交換格式,下面這篇文章主要給大家介紹了關(guān)于Python中使用json.load()和json.loads()加載json數(shù)據(jù)的方法實(shí)例,文中通過實(shí)例代碼介紹的非常詳細(xì),需要的朋友可以參考下2022-08-08
OneFlow源碼解析之Eager模式下Tensor存儲管理
這篇文章主要為大家介紹了OneFlow源碼解析之Eager模式下Tensor的存儲管理實(shí)現(xiàn)示例詳解,有需要的朋友可以借鑒參考下,希望能夠有所幫助,祝大家多多進(jìn)步,早日升職加薪2023-04-04
詳解如何利用pandas進(jìn)行數(shù)據(jù)行轉(zhuǎn)列和列轉(zhuǎn)行
這篇文章主要為大家詳細(xì)介紹了如何利用pandas進(jìn)行數(shù)據(jù)行轉(zhuǎn)列和列轉(zhuǎn)行,文中的示例代碼講解詳細(xì),感興趣的小伙伴可以跟隨小編一起學(xué)習(xí)一下2007-02-02
Python利用watchdog模塊監(jiān)控文件變化
這篇文章主要為大家介紹一個Python中的模塊:watchdog模塊,它可以實(shí)現(xiàn)監(jiān)控文件的變化。文中通過示例詳細(xì)介紹了watchdog模塊的使用,需要的可以參考一下2022-06-06

