問題描述
前段時間遇到個問題,自己內部系統調用出現重復請求導致數據混亂。
發生條件:接受到一個請求,該請求沒有執行完成又接受到相同請求,導致數據錯誤(如果是前一個請求執行完成,馬上又接受相同請求不會有問題)
問題分析:是由于數據庫的臟讀導致
問題解決思路
1.加一把大大的鎖 (是最簡單的實現方式,但是性能堪憂,而且會阻塞請求)
2.實現請求攔截 (可以共用,但是怎么去實現卻是一個問題,怎么用一個優雅的方式實現,并且方便復用)
3.修改實現 (會對原有代碼做改動,存在風險,最主要的是不能共用)
最終實現方式
通過注解+spring AOP 的方式實現
使用
通過在任意方法上添加注解NotDuplicate
類1:
1
2
3
4
5
6
7
8
9
10
11
12
|
import static java.lang.annotation.ElementType.METHOD; import java.lang.annotation.Documented; import java.lang.annotation.Retention; import java.lang.annotation.RetentionPolicy; import java.lang.annotation.Target; @Target ({METHOD}) @Retention (RetentionPolicy.RUNTIME) @Documented public @interface NotDuplicate { } |
類2:
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
|
import java.lang.reflect.Method; import java.util.Set; import java.util.concurrent.ConcurrentSkipListSet; import org.aspectj.lang.ProceedingJoinPoint; import org.aspectj.lang.annotation.Around; import org.aspectj.lang.annotation.Aspect; import org.aspectj.lang.annotation.Pointcut; import org.aspectj.lang.reflect.MethodSignature; import org.springframework.stereotype.Component; @Aspect @Component public class NotDuplicateAop { private static final Set<String> KEY = new ConcurrentSkipListSet<>(); @Pointcut ( "@annotation(com.hhly.skeleton.base.filter.NotDuplicate)" ) public void duplicate() { } /** * 對方法攔截后進行參數驗證 * @param pjp * @return * @throws Throwable */ @Around ( "duplicate()" ) public Object duplicate(ProceedingJoinPoint pjp) throws Throwable { MethodSignature msig = (MethodSignature) pjp.getSignature(); Method currentMethod = pjp.getTarget().getClass().getMethod(msig.getName(), msig.getParameterTypes()); //拼接簽名 StringBuilder sb = new StringBuilder(currentMethod.toString()); Object[] args = pjp.getArgs(); for (Object object : args) { if (object != null ){ sb.append(object.getClass().toString()); sb.append(object.toString()); } } String sign = sb.toString(); boolean success = KEY.add(sign); if (!success){ throw new ServiceRuntimeException( "該方法正在執行,不能重復請求" ); } try { return pjp.proceed(); } finally { KEY.remove(sign); } } } |
以上就是本次給大家講述的全部內容以及相關代碼,如果大家還有任何問題可以在下方的留言區討論,感謝大家對服務器之家的支持。
原文鏈接:https://blog.csdn.net/u010704600/article/details/77799169