spring层面处理网络抖动致使的重复写入数据,实现请求的幂等性

问题描述

在用户使用网站建立新闻时,因为网络波动,致使发送了多个建立新闻的请求,使得系统中存在了冗余的数据。java

问题分析

  1. 看过不少文章,大部分思路都是:若是同一用户在很短期内发送了重复的post请求,那么后台只处理第一个请求,后面的请求则过滤掉。
  2. 具体实现方式则是添加一个springmvc的拦截器,当一个post请求过来时,首先去缓存中查询短期内有没有相同请求到来,若是没有,则把该请求放入缓存,并将请求传递下去;若是有,则再也不执行后续操做。
  3. 可是这种方式也会面临一个问题:若是两个请求同时到来,同时读缓存,这样两个请求都不会读到记录,致使重复执行了两个请求。
  4. 因而咱们想到给缓存加锁,这样就能够避免3中出现的状况。可是众所周知,普通的加锁是很影响效率的,咱们也不能舍弃效率来保证幂等性。为了提高加锁后的效率,我采用了DCL(双重检查锁,没有学过的小伙伴能够去查一查)来进行缓存的加锁读写。

代码

因为个人应用是单例的,因此采用guava来作缓存。web

import com.csdc.cett.exception.RequestException;
import com.csdc.cett.util.MD5;
import com.google.common.cache.Cache;
import com.google.common.cache.CacheBuilder;
import org.springframework.stereotype.Component;
import org.springframework.web.servlet.HandlerInterceptor;
import org.springframework.web.servlet.ModelAndView;

import javax.servlet.http.HttpServletRequest;
import javax.servlet.http.HttpServletResponse;
import java.util.Enumeration;
import java.util.concurrent.TimeUnit;

/**
 * 处理重复请求,保持请求的幂等性
 * 使用双重检查锁定(DCL)
 *
 * @author ksyzz
 * @since <pre>2019/06/20</pre>
 */
@Component
public class RequestInterceptor implements HandlerInterceptor {

    private static volatile Cache<String, String> loadingCache = CacheBuilder.newBuilder()
            .maximumSize(500)
            .expireAfterWrite(5, TimeUnit.SECONDS)
            .build();

    @Override
    public boolean preHandle(HttpServletRequest request, HttpServletResponse response, Object handler) throws Exception {
        // 只须要保持用户登陆后的POST请求的幂等性,此处要排除文件分片上传
        if ("POST".equals(request.getMethod()) && request.getHeader("Auth-Token") != null && !request.getRequestURI().contains("/upload/files")) {
            // 要求:用户5秒内不能重复提交相同url,相同参数的请求
            // 存储方式为 md5(URI+Auth-Token+RequestParams+InputStream)
            StringBuilder md5 = new StringBuilder();
            // Auth-Token为用户身份标识
            String token = request.getHeader("Auth-Token");
            md5.append(request.getRequestURI());
            md5.append(token);
            Enumeration<String> parameterNames = request.getParameterNames();
            while (parameterNames.hasMoreElements()) {
                String key = parameterNames.nextElement();
                String parameter = request.getParameter(key);
                md5.append("&").append(key).append("=").append(parameter);
            }
            String cacheKey = MD5.getHashString(md5.toString());

            // 校验是否已经提交过请求

            if (loadingCache.getIfPresent(cacheKey) == null) {
                synchronized (loadingCache) {
                    if (loadingCache.getIfPresent(cacheKey) == null) {
                        // 此处value可不设置
                        loadingCache.put(cacheKey, "");
                        return true;
                    }
                }
            }
            throw new RequestException("请勿重复提交");
        }
        return true;
    }

    @Override
    public void postHandle(HttpServletRequest request, HttpServletResponse response, Object handler, ModelAndView modelAndView) throws Exception {

    }

    @Override
    public void afterCompletion(HttpServletRequest request, HttpServletResponse response, Object handler, Exception ex) throws Exception {

    }

}
相关文章
相关标签/搜索