java并發(fā)訪問重復(fù)請求過濾問題
問題描述
前段時間遇到個問題,自己內(nèi)部系統(tǒng)調(diào)用出現(xiàn)重復(fù)請求導(dǎo)致數(shù)據(jù)混亂。
發(fā)生條件:接受到一個請求,該請求沒有執(zhí)行完成又接受到相同請求,導(dǎo)致數(shù)據(jù)錯誤(如果是前一個請求執(zhí)行完成,馬上又接受相同請求不會有問題)
問題分析:是由于數(shù)據(jù)庫的臟讀導(dǎo)致
問題解決思路
1.加一把大大的鎖 (是最簡單的實(shí)現(xiàn)方式,但是性能堪憂,而且會阻塞請求)
2.實(shí)現(xiàn)請求攔截 (可以共用,但是怎么去實(shí)現(xiàn)卻是一個問題,怎么用一個優(yōu)雅的方式實(shí)現(xiàn),并且方便復(fù)用)
3.修改實(shí)現(xiàn) (會對原有代碼做改動,存在風(fēng)險,最主要的是不能共用)
最終實(shí)現(xiàn)方式
通過注解+spring AOP 的方式實(shí)現(xiàn)
使用
通過在任意方法上添加注解NotDuplicate
類1:
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:
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() { } /** * 對方法攔截后進(jìn)行參數(shù)驗(yàn)證 * @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("該方法正在執(zhí)行,不能重復(fù)請求"); } try { return pjp.proceed(); } finally { KEY.remove(sign); } } }
以上就是本次給大家講述的全部內(nèi)容以及相關(guān)代碼,如果大家還有任何問題可以在下方的留言區(qū)討論,感謝大家對腳本之家的支持。
相關(guān)文章
基于spring實(shí)現(xiàn)websocket實(shí)時推送實(shí)例
這篇文章主要為大家詳細(xì)介紹了基于spring實(shí)現(xiàn)websocket實(shí)時推送實(shí)例,具有一定的參考價值,感興趣的小伙伴們可以參考一下2018-03-03從Android源碼剖析Intent查詢匹配的實(shí)現(xiàn)
這篇文章主要介紹了從Android源碼剖析Intent查詢匹配的實(shí)現(xiàn),Intent部分的源碼為Java代碼,需要的朋友可以參考下2015-07-07關(guān)于struts2中Action名字的大小寫問題淺談
這篇文章主要給大家介紹了關(guān)于struts2中Action名字大小寫問題的相關(guān)資料,文中介紹的非常詳細(xì),對大家具有一定的參考學(xué)習(xí)價值,需要的朋友們下面跟著小編一起來學(xué)習(xí)學(xué)習(xí)吧。2017-06-06基于idea把maven工程轉(zhuǎn)換為web項(xiàng)目
這篇文章主要介紹了基于idea把maven工程轉(zhuǎn)換為web項(xiàng)目,文中通過示例代碼介紹的非常詳細(xì),對大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價值,需要的朋友可以參考下2020-04-04Java中實(shí)體類為什么要實(shí)現(xiàn)Serializable序列化的作用
這篇文章主要介紹了Java中實(shí)體類為什么要實(shí)現(xiàn)Serializable序列化的作用,文中通過示例代碼介紹的非常詳細(xì),對大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價值,需要的朋友們下面隨著小編來一起學(xué)習(xí)學(xué)習(xí)吧2020-11-11