X5内核 请求超时后会自动阻止请求返回并由代理服务器将原参数重新发送请求到服务层代码。但由于第一次请求已经请求到服务器,会导致出现重复下单、支付等重大问题。
该问题由于腾讯x5浏览器会自动阻止第一次请求返回到页面,届时将抛出io异常。最开始打算通过拦截器来进行拦截第二次请求,但这样将使页面无法接受到返回信息报错。
初步解决思路 当第二次请求访问进服务层时暂停该线程,并定时循环查询第一次请求是否返回成功,将第一次返回成功的结果赋值给第二次。
由于我们系统中所有页面请求都会过一个通用controller 所以也会有很多正常请求通过 为了不影响整体使用
这里我用到了页面验签的方法,用于验证请求是否为同一次。这里先将参数进行排序,再用md5加密,添加一个“sign”的参数到后台,后台再排序加密进行比对,进行验签,注意这里参数加入了
时间戳,用于防止重复提交
KGF.getMD5 = function(params){ params["key"] = "ouE6yWvy"; var sParams = Object.keys(params).sort(); var oriString = ""; for(key in sParams){ oriString += sParams[key]+"="+params[sParams[key]]+"&"; } oriString = oriString.substr(0,oriString.length-1); delete params.key; return KGF.md5(oriString); };
验签成功后,由于需要将第一次返回成功的结果赋值给第二次。所以需要将调用接口成功的结果存入seesion中,但此时需要及时去清理,否则容易造成服务器内存溢出。
默认调用接口成功后就将结果集放入session,验签一通过就去取,这里的关键在于何时去删除session中的值。由于所有结果访问均会通过该控制器,为了区分不同的请求,
故用sign作为键,返回结果为值。
下面贴出代码
@RequestMapping(value={"/common/ajax.do"}, method={RequestMethod.POST}, produces = MediaType.APPLICATION_JSON_VALUE) @ResponseBody public List<AjaxResponse> execute2(@RequestHeader(value="Accept", required=false) String accept, HttpServletRequest request){ XMLResults results = new XMLResults(); try{ AjaxObject object = (AjaxObject)request.getAttribute("ParamName"); String openid = (String) request.getSession().getAttribute("openid"); Class[] cArg = new Class[2]; cArg[0] = HttpServletRequest.class; cArg[1] = Map.class; DataResult dataResult = new DataResult(); Map<String, Object> params = object.getParams(); String signFromAjax = (String) params.get("sign"); long startTime = 0; long endTime = 0; if(!checkSign(params)){ log.error("接口请求验签失败"); dataResult.setRetInfo("-1", "接口请求验签失败", CommonUtil.parseDoubleList(new HashMap())); }else{ String signFromSession = (String) request.getSession().getAttribute(signFromAjax); request.getSession().setAttribute(signFromAjax,signFromAjax); //处理腾讯安卓微信浏览器,10s重发请求的问题 if(signFromSession != null){ for(int i = 0; i <10; i++){ try { Thread.currentThread().sleep(3000); } catch (InterruptedException e) { e.printStackTrace(); } log.debug("====================重复请求读取接口返回信息:第"+(i+1)+"次===================="); DataResult dataResultFromSession = (DataResult) request.getSession().getAttribute(signFromAjax+"interfaceResult"); if(dataResultFromSession != null){ request.getSession().removeAttribute(signFromAjax); request.getSession().removeAttribute(signFromAjax+"interfaceResult"); results = this.baseService.convertResult(dataResult); dataResult = dataResultFromSession; break; } if(i == 9){ dataResult.setRetInfo("-1", "接口请求超时,请联系管理员", CommonUtil.parseDoubleList(new HashMap())); } } }else{ startTime = new Date().getTime(); params.put("requestIp", CommonUtil.getIpAddr(request)); params.put("openid", openid); params.put("deviceType", CommonUtil.getDeviceType(request)); params.put("browserType", CommonUtil.getBrowserType(request)); params.put("g_stationaddr", CommonUtil.getIpAddr(request)); Method method = commonDataModel.getClass().getDeclaredMethod(object.getCode(), cArg); dataResult = (DataResult) method.invoke(commonDataModel, request, params); endTime = new Date().getTime(); request.getSession().setAttribute(signFromAjax+"interfaceResult", dataResult); } } if(endTime - startTime <= 10000){ request.getSession().removeAttribute(signFromAjax); request.getSession().removeAttribute(signFromAjax+"interfaceResult"); } results = this.baseService.convertResult(dataResult); }catch (Exception e){ log.error("数据请求失败,错误信息:" + e.getMessage()); results.setRetInfo("-1", "数据请求失败"); } return webHelper.convertAjaxResponse(results); }