溫馨提示×

您好,登錄后才能下訂單哦!

密碼登錄×
登錄注冊(cè)×
其他方式登錄
點(diǎn)擊 登錄注冊(cè) 即表示同意《億速云用戶服務(wù)條款》

怎么解決java并發(fā)請(qǐng)求下數(shù)據(jù)插入重復(fù)問(wèn)題

發(fā)布時(shí)間:2021-11-12 15:22:27 來(lái)源:億速云 閱讀:150 作者:iii 欄目:開(kāi)發(fā)技術(shù)

本篇內(nèi)容介紹了“怎么解決java并發(fā)請(qǐng)求下數(shù)據(jù)插入重復(fù)問(wèn)題”的有關(guān)知識(shí),在實(shí)際案例的操作過(guò)程中,不少人都會(huì)遇到這樣的困境,接下來(lái)就讓小編帶領(lǐng)大家學(xué)習(xí)一下如何處理這些情況吧!希望大家仔細(xì)閱讀,能夠?qū)W有所成!

前言

前段時(shí)間發(fā)現(xiàn)數(shù)據(jù)庫(kù)里經(jīng)常會(huì)存在兩條相同的用戶數(shù)據(jù),導(dǎo)致數(shù)據(jù)查詢異常。查了原因,發(fā)現(xiàn)前端微信小程序在授權(quán)登錄時(shí),有時(shí)會(huì)出現(xiàn)同時(shí)發(fā)送了兩條一模一樣的請(qǐng)求(也就是常說(shuō)的并發(fā))。雖然后端代碼有做防重復(fù)的判斷,但是避免不了并發(fā)時(shí)候的重復(fù)性操作。于是就開(kāi)始考慮并發(fā)的解決方案,解決方案有很多,從攔截請(qǐng)求到數(shù)據(jù)庫(kù)層面都可以入手。

我們采用了對(duì)請(qǐng)求報(bào)文生成摘要信息+Redis分布式鎖的方案。運(yùn)行了一段時(shí)間,功能很可靠,代碼也很簡(jiǎn)潔。于是上來(lái)做下記錄以便后續(xù)參考。

解決方案說(shuō)明:

系統(tǒng)架構(gòu)用的Spring boot,定義一個(gè)Filter過(guò)濾器對(duì)請(qǐng)求進(jìn)行過(guò)濾,然后對(duì)請(qǐng)求報(bào)文生成摘要信息并設(shè)置Redis分布式鎖。通過(guò)摘要和鎖判斷是否為同一請(qǐng)求。

分布式鎖工具類

public class ContextLJ {
	
	private static final Integer JD = 0;
	
	  /**
	   * 上鎖 使用redis 為分布式項(xiàng)目 加鎖
	   * @param sign
	   * @param tiD
	   * @return
	   * @throws Exception
	   */
	  public static boolean lock(String sign, String tiD) {
	    synchronized (JD) { // 加鎖
	    	Cache<String> cache = CacheManager.getCommonCache(sign);
	    	if(cache == null || StringUtils.isBlank(cache.getValue())) {
	    		CacheManager.putCommonCacheInfo(sign, tiD, 10000);
	    		return true;
			}
	    	return false;
	    }
	 }
	 
	  /**
	   * 鎖驗(yàn)證
	   * @param sign
	   * @param tiD
	   * @return
	   */
	  public static boolean checklock(String sign, String tiD){
		  Cache<String> cache = CacheManager.getCommonCache(sign);
		  String uTid = StringUtils.replace(cache.getValue(), "\"", "");
		  return tiD.equals(uTid);
	  }
	 
	  /**
	   * 去掉鎖
	   * @param sign
	   * @param tiD
	   */
	  public static void clent (String sign, String tiD){
		    if (checklock(sign, tiD)) {
		    	CacheManager.clearOnly(sign);
		    }
	  }
	 
	  /**
	   * 獲取摘要
	   * @param request
	   */
	  public static String getSign(ServletRequest request){
	    // 此工具是將 request中的請(qǐng)求內(nèi)容 拼裝成 key=value&key=value2 的形式 源碼在線面
	    String sign = null;
	    try {
	    	Map<String, String> map =  getRequstMap((HttpServletRequest) request);
	    	// 生成摘要
	    	sign = buildRequest(map);
	    } catch (Exception e) {
	    	e.printStackTrace();
	    }
	    return sign;
	  }
	  
	  public static Map<String, String> getRequstMap(HttpServletRequest req) throws Exception{
 		    Map<String,String> params = new HashMap<String,String>();
 		    params.put("uri", req.getRequestURI());
		    Map<String, String[]> requestParams = req.getParameterMap();
		    for (Iterator<String> iter = requestParams.keySet().iterator(); iter.hasNext();) {
		      String name = (String) iter.next();
		      String[] values = (String[]) requestParams.get(name);
		      String valueStr = "";
		      for (int i = 0; i < values.length; i++) {
		        valueStr = (i == values.length - 1) ? valueStr + values[i] : valueStr + values[i] + ",";
		      }
		      params.put(name, valueStr);
		    }
		    return params;
	}
	  
	 private static String buildRequest(Map<String, String> map) {
		 List<String> signList = new ArrayList<>();
		 for(Entry<String, String> entry : map.entrySet()) {
			 signList.add(entry.getKey() + "=" + entry.getValue());
		 }
		 String sign = StringUtils.join(signList, "&");
		 return DigestUtils.md5Hex(sign);
	}
	
}

在過(guò)濾器實(shí)現(xiàn)請(qǐng)求攔截

/**
 * 過(guò)濾頻繁請(qǐng)求
 */
@Slf4j
@Component
public class MyFilter implements Filter{
	
	@Override
	public void init(FilterConfig filterConfig) throws ServletException {
		
	}

	@Override
	public void doFilter(ServletRequest request, ServletResponse myResp, FilterChain chain) throws IOException, ServletException {
		HttpServletRequest req = (HttpServletRequest) request;
		Boolean isDict = StringUtils.contains(req.getRequestURI(), "/dict/getDatas");
		Boolean isFile = StringUtils.contains(req.getRequestURI(), "/files/file");
		if(isDict || isFile) {
			chain.doFilter(request, myResp); // 查詢數(shù)據(jù)字典或者文件,直接放行
			return;
		}
		String sign = "sign_" + ContextLJ.getSign(request); // 生成摘要
	    String tiD = RandomUtils.randomCode(3) + "_" + Thread.currentThread().getId(); // 當(dāng)前線程的身份
	    try { 
	    	if (!ContextLJ.lock(sign, tiD)) {
	    		Map<String,String> map = ContextLJ.getRequstMap((HttpServletRequest)request);
	    		log.warn("放棄相同并發(fā)請(qǐng)求【" + sign+ "】【" + tiD+"】"+JSON.toJSONString(map));
	    		frequentlyError(myResp);
	    		return;
	    	}
	    	if (!ContextLJ.checklock(sign, tiD)) {
	    		Map<String,String> map = ContextLJ.getRequstMap((HttpServletRequest)request);
		    	  log.warn("加鎖驗(yàn)證失敗 【" + sign+ "】【" + tiD+"】"+JSON.toJSONString(map));
		    	  frequentlyError(myResp);
		    	  return;
	    	}
	    	chain.doFilter(request, myResp); // 放行
	    } catch (Exception e) { // 捕獲到異常 進(jìn)行異常過(guò)濾
		      log.error("", e);
		      myResp.getWriter().write(JSON.toJSONString(ApiRs.asError("服務(wù)器繁忙,請(qǐng)重試")));
	    } finally {
	    	ContextLJ.clent(sign, tiD);
	    }
	}

	@Override
	public void destroy() {
		
	}
	
	/**
	 * 頻繁請(qǐng)求
	 */
	private void frequentlyError(ServletResponse myResp) throws IOException {
	  ((HttpServletResponse) myResp).setHeader("Content-type", "text/html;charset=UTF-8");
	  myResp.getWriter().write(JSON.toJSONString(ApiRs.asError("稍安勿躁,不要頻繁請(qǐng)求")));
	}

}

“怎么解決java并發(fā)請(qǐng)求下數(shù)據(jù)插入重復(fù)問(wèn)題”的內(nèi)容就介紹到這里了,感謝大家的閱讀。如果想了解更多行業(yè)相關(guān)的知識(shí)可以關(guān)注億速云網(wǎng)站,小編將為大家輸出更多高質(zhì)量的實(shí)用文章!

向AI問(wèn)一下細(xì)節(jié)

免責(zé)聲明:本站發(fā)布的內(nèi)容(圖片、視頻和文字)以原創(chuàng)、轉(zhuǎn)載和分享為主,文章觀點(diǎn)不代表本網(wǎng)站立場(chǎng),如果涉及侵權(quán)請(qǐng)聯(lián)系站長(zhǎng)郵箱:is@yisu.com進(jìn)行舉報(bào),并提供相關(guān)證據(jù),一經(jīng)查實(shí),將立刻刪除涉嫌侵權(quán)內(nèi)容。

AI