SpringBoot統(tǒng)計(jì)一個(gè)Bean中方法的調(diào)用次數(shù)的實(shí)現(xiàn)步驟
實(shí)現(xiàn)思路
通過(guò)AOP即可實(shí)現(xiàn),通過(guò)AOP對(duì)Bean進(jìn)行代理,在每次執(zhí)行方法前或者后進(jìn)行幾次計(jì)數(shù)統(tǒng)計(jì)。這個(gè)主要就是考慮好如何避免并發(fā)情況下不準(zhǔn),以及如何使用AOP實(shí)現(xiàn)代理。
前置條件
首先搭建一個(gè)spring boot工程,我這里用的是3x版本
搭建步驟:
新版idea創(chuàng)建springboot項(xiàng)目
新版idea創(chuàng)建spring boot項(xiàng)目的詳細(xì)教程_java_腳本之家 (jb51.net)
導(dǎo)入依賴:
pom.xml:
<dependency> <groupId>org.springframework.boot</groupId> <artifactId>spring-boot-starter-aop</artifactId> </dependency>
實(shí)現(xiàn)步驟
首先我們先自定義一個(gè)注解
有了這個(gè)注解之后,我們可以在想要統(tǒng)計(jì)的方法上加上這個(gè)注解
名稱隨便起但要見(jiàn)名知意
代碼如下:
import java.lang.annotation.ElementType; import java.lang.annotation.Retention; import java.lang.annotation.RetentionPolicy; import java.lang.annotation.Target; /** * @author mijiupro */ @Retention(RetentionPolicy.RUNTIME)// 指定注解的保留策略為RUNTIME @Target(ElementType.METHOD)// 指定該注解可以用于方法 public @interface MethodCallCount { }
接下來(lái)定義一個(gè)切面
通過(guò)該切面來(lái)對(duì)這個(gè)注解進(jìn)行增強(qiáng)處理
代碼如下:
import lombok.extern.slf4j.Slf4j; import org.aspectj.lang.ProceedingJoinPoint; import org.aspectj.lang.annotation.Around; import org.aspectj.lang.annotation.Aspect; import org.springframework.stereotype.Component; import java.util.Map; import java.util.concurrent.ConcurrentHashMap; import java.util.concurrent.atomic.AtomicInteger; /** * @author mijiupro */ @Aspect// 聲明這是一個(gè)切面 @Component// 聲明這是一個(gè)Spring Bean(交給Spring管理) @Slf4j public class MethodCallCountAspect { // 用于存儲(chǔ)方法調(diào)用次數(shù)的Map,使用ConcurrentMap保證線程安全 private final Map<String, AtomicInteger> counterMap = new ConcurrentHashMap<>(); @Around("@annotation(com.mijiu.commom.aop.annotation.MethodCallCount)") public Object methodCallCountAspect(ProceedingJoinPoint joinPoint) { String methodName = joinPoint.getSignature().toShortString(); try{ return joinPoint.proceed(); }catch (Throwable ignored){ //異常處理 return null; }finally { AtomicInteger counter = counterMap.computeIfAbsent(methodName,k -> new AtomicInteger(0)); counter.incrementAndGet(); log.info("方法 {} 調(diào)用次數(shù):{}", methodName, counter.get()); } } // 提供一個(gè)方法,用于獲取方法調(diào)用次數(shù)的Map public Map<String, AtomicInteger> getCounterMap() { return new ConcurrentHashMap<>(counterMap); } }
需要統(tǒng)計(jì)方法上使用該注解
有了以上注解和切面后,只需要在我們想要統(tǒng)計(jì)的方法上使用該注解就行了
測(cè)試
啟動(dòng)項(xiàng)目調(diào)用一下接口
但是需要注意的是,這個(gè)統(tǒng)計(jì)結(jié)果只在內(nèi)存中有效,如果應(yīng)用發(fā)生重啟,就會(huì)歸零了。如果想要持久化保存,就需要考慮持久化存儲(chǔ)了,如存在mysql或者redis中。
另外,如果并發(fā)特別高,對(duì)統(tǒng)計(jì)結(jié)果要求沒(méi)那么精確,可以用LongAdder替代AtomicInteger
以上就是SpringBoot統(tǒng)計(jì)一個(gè)Bean中方法的調(diào)用次數(shù)的實(shí)現(xiàn)步驟的詳細(xì)內(nèi)容,更多關(guān)于SpringBoot統(tǒng)計(jì)Bean調(diào)用次數(shù)的資料請(qǐng)關(guān)注腳本之家其它相關(guān)文章!
相關(guān)文章
SpringBoot使用Caffeine實(shí)現(xiàn)緩存的示例代碼
本文主要介紹了SpringBoot使用Caffeine實(shí)現(xiàn)緩存的示例代碼,文中通過(guò)示例代碼介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友們下面隨著小編來(lái)一起學(xué)習(xí)學(xué)習(xí)吧2022-07-07MongoDB中ObjectId的誤區(qū)及引起的一系列問(wèn)題
這篇文章主要介紹了MongoDB中ObjectId的誤區(qū)及引起的一系列問(wèn)題,非常不錯(cuò),具有參考借鑒價(jià)值,需要的朋友可以參考下2016-12-12參數(shù)校驗(yàn)Spring的@Valid注解用法解析
這篇文章主要介紹了參數(shù)校驗(yàn)Spring的@Valid注解用法,具有很好的參考價(jià)值,希望對(duì)大家有所幫助。如有錯(cuò)誤或未考慮完全的地方,望不吝賜教2021-08-08使用SpringMVC訪問(wèn)Controller接口返回400BadRequest
這篇文章主要介紹了使用SpringMVC訪問(wèn)Controller接口返回400BadRequest,具有很好的參考價(jià)值,希望對(duì)大家有所幫助。如有錯(cuò)誤或未考慮完全的地方,望不吝賜教2022-03-03JDK動(dòng)態(tài)代理接口和接口實(shí)現(xiàn)類深入詳解
這篇文章主要介紹了JDK動(dòng)態(tài)代理接口和接口實(shí)現(xiàn)類,JDK動(dòng)態(tài)代理是代理模式的一種實(shí)現(xiàn)方式,因?yàn)樗腔诮涌趤?lái)做代理的,所以也常被稱為接口代理,文中通過(guò)實(shí)例代碼介紹的非常詳細(xì),需要的朋友可以參考下2022-06-06