SpringAop中的Advice通知實(shí)例
一 概述
- @Before:前置通知
- @After :后置通知,方法執(zhí)行完之后
- @AfterReturning:返回通知,完成執(zhí)行之后
- @AfterThrowing:異常通知,拋出異常之后
- @Around:環(huán)繞通知
二 Advice應(yīng)用實(shí)例
通過環(huán)繞通知獲取方法執(zhí)行時(shí)間
@Component
@Aspect
public class RepositoryAspect {
@Pointcut("execution(* com.repository..*(..))")
private void crud() {
}
//環(huán)繞通知
@Around("crud()")
public Object logPerformance(ProceedingJoinPoint joinPoint) throws Throwable {
long startTimes = System.currentTimeMillis();
Object result = null;
String name = "-";
try {
name = joinPoint.getSignature().toShortString();
result = joinPoint.proceed();
return result;
} catch (Throwable t) {
throw t;
} finally {
long duration = System.currentTimeMillis() - startTimes;
StringBuilder stringBuilder = new StringBuilder();
stringBuilder.append(name);
System.out.println(stringBuilder+"執(zhí)行時(shí)間為"+duration)
}
}
}public class AbstractAspect {
protected ThreadLocal<Map<String, Object>> threadLocal = new ThreadLocal<>();
protected ThreadLocal<Map<String, Object>> getThreadLocal(){
if (threadLocal.get() == null){
threadLocal.set(new HashMap<>());
}
return threadLocal;
}
}@Component
@Aspect
public class ControllerAspect extends AbstractAspect {
//對于controller包下所有類和方法進(jìn)行獲取,除被注解NoHealth標(biāo)注的類除外。
@Pointcut("execution(* com.controller..*(..))&&!@annotation(NoHealth)")
private void controllerPointCut() {
}
//在方法執(zhí)行之前對參數(shù)進(jìn)行處理
@Before(value = "controllerPointCut()")
public void doBefore(JoinPoint joinPoint) {
try {
getThreadLocal().get().put(joinPoint.getSignature().toString(), System.currentTimeMillis());
Object[] args = joinPoint.getArgs();
MethodSignature methodSignature = (MethodSignature) joinPoint.getSignature();
String[] parameterNamesArray = methodSignature.getParameterNames();
int i = 0;
List<String> argsList = new ArrayList<>();
if (!ObjectUtils.isEmpty(args)) {
for (Object object : args) {
if (ObjectUtils.isEmpty(object)
|| (object instanceof BeanPropertyBindingResult)
|| (object instanceof HttpServletResponse)
|| (object instanceof Boolean)
|| (object instanceof String)
|| (object instanceof Integer)) {
if ((object instanceof Boolean) || (object instanceof String) || (object instanceof Integer)) {
StringBuilder stringBuilder = new StringBuilder();
stringBuilder.append("{\"");
stringBuilder.append(parameterNamesArray[i]);
stringBuilder.append("\":\"");
stringBuilder.append(object);
stringBuilder.append("\"}");
argsList.add(JSONObject.fromObject(String.valueOf(stringBuilder)).toString());
}
i++;
continue;//跳出當(dāng)前循環(huán),且不執(zhí)行此語句后面的語句!
}
if (!(object instanceof HttpServletRequest)) {
argsList.add(JSONObject.fromObject(object).toString());
} else {
HttpServletRequest request = (HttpServletRequest) object;
Enumeration<String> parameterNames = request.getParameterNames();
Map<String, String> argsMap = new HashMap<>();
while (parameterNames.hasMoreElements()) {
String elementName = parameterNames.nextElement();
argsMap.put(elementName, request.getAttribute(elementName).toString());
}
if (!ObjectUtils.isEmpty(argsMap)) {
argsList.add(JSONObject.fromObject(argsMap).toString());
}
}
i++;
}
}
StringBuilder stringBuilder = new StringBuilder();
stringBuilder.append(JSONArray.fromObject(argsList));
} catch (Exception e) {
}
}
//對方法執(zhí)行之后的返回值進(jìn)行某種操作
@AfterReturning(value = "controllerPointCut()", returning = "response")
public void responseLog(JoinPoint joinPoint, Object response) {
try {
if (response == null) {
response = new Object();
}
long duration = System.currentTimeMillis() - (Long) threadLocal.get().get(joinPoint.getSignature().toString());
StringBuilder stringB uilder = new StringBuilder();
stringBuilder.append(JSONObject.fromObject(response));
} catch (Exception e) {
}
}
//對方法拋出的異常進(jìn)行操作
@AfterThrowing(value = "controllerPointCut()", throwing = "e")
public void exceptionLog(Exception e) {
}
}根據(jù)這些實(shí)例就可以參照寫一些簡單的SpringAOP實(shí)例了!
到此這篇關(guān)于SpringAop中的Advice通知詳解的文章就介紹到這了,更多相關(guān)Advice通知內(nèi)容請搜索腳本之家以前的文章或繼續(xù)瀏覽下面的相關(guān)文章希望大家以后多多支持腳本之家!
相關(guān)文章
springboot使用外置tomcat啟動(dòng)方式
這篇文章主要介紹了springboot使用外置tomcat啟動(dòng)方式,具有很好的參考價(jià)值,希望對大家有所幫助。如有錯(cuò)誤或未考慮完全的地方,望不吝賜教2021-11-11
SpringBoot Event實(shí)現(xiàn)異步消費(fèi)機(jī)制的示例代碼
這篇文章主要介紹了SpringBoot Event實(shí)現(xiàn)異步消費(fèi)機(jī)制,ApplicationEvent以及Listener是Spring為我們提供的一個(gè)事件監(jiān)聽、訂閱的實(shí)現(xiàn),內(nèi)部實(shí)現(xiàn)原理是觀察者設(shè)計(jì)模式,文中有詳細(xì)的代碼示例供大家參考,需要的朋友可以參考下2024-04-04
springboot2整合redis使用lettuce連接池的方法(解決lettuce連接池?zé)o效問題)
這篇文章主要介紹了springboot2整合redis使用lettuce連接池(解決lettuce連接池?zé)o效問題),本文給大家介紹的非常詳細(xì),對大家的學(xué)習(xí)或工作具有一定的參考借鑒價(jià)值,需要的朋友可以參考下2020-12-12
利用Java實(shí)現(xiàn)網(wǎng)站聚合工具
互聯(lián)網(wǎng)上有數(shù)以萬億計(jì)的網(wǎng)站,每個(gè)網(wǎng)站大都具有一定的功能。搜索引擎雖然對互聯(lián)網(wǎng)上的部分網(wǎng)站建立了索引,但是其作為一個(gè)大而全的搜索系統(tǒng),無法很好的定位到一些特殊的需求。因此本文將介紹一個(gè)用java實(shí)現(xiàn)的網(wǎng)站數(shù)據(jù)聚合工具,需要的可以參考一下2022-01-01
Java實(shí)現(xiàn)文件上傳至服務(wù)器的方法
這篇文章主要為大家詳細(xì)介紹了Java實(shí)現(xiàn)文件上傳至服務(wù)器的方法,文中示例代碼介紹的非常詳細(xì),具有一定的參考價(jià)值,感興趣的小伙伴們可以參考一下2018-01-01

