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

