request.getParameterMap()返回的Map锁定问题

本文详细记录了解决使用request.getParameterMap()时遇到的异常问题的过程,包括理解ParameterMap类的加锁机制、代码示例及最终使用反射解决方法。

最近做项目,用到request.getParameterMap();在对返回的Map进行操作的时候产生异常

Cannot find message associated with key parameterMap.locked,

网上找了一下解决方法,记录一下,希望对产生同样问题的朋友有所帮助,如下:

 

  1. Map requestParams=request.getParameterMap();  
  2. ......  
  3. requestParams.remove("key" );  

运行时会报错误:

Cannot find message associated with key parameterMap.locked

at org.apache.catalina.util.ParameterMap.remove(ParameterMap.java:204)

从输出的错误信息发现,原来request.getParameterMap();返回的不是一般的Map,而是org.apache.catalina.util.ParameterMap!

反编译ParameterMap,发现其继承自HashMap:

public final class ParameterMap extends HashMap

其中有一个加锁的方法,问题就出在这里了: 

 

  1. public   void  setLocked( boolean  locked)  
  2. {  
  3.     this .locked = locked;  
  4. }  

其他所有修改内容的方法,都会判断是否上锁。如果处于锁定状态,就会抛出异常,比如最常用的put方法:

 

  1. public  Object put(Object key, Object value)  
  2. {  
  3.     if (locked)  
  4.         throw   new  IllegalStateException(sm.getString( "parameterMap.locked" ));  
  5.     else   
  6.         return   super .put(key, value);  
  7. }  

原来如此!

于是,从Tomcat下拷来了catalina.jar,代码写成

 

  1. ParameterMap requestParams=(ParameterMap)request.getParameterMap();  
  2. requestParams.setLocked(false );  
  3. requestParams.remove("key" );  
  4. ......  

运行时还是报类型转换错误:不能从org.apache.catalina.util.ParameterMap转换为org.apache.catalina.util.ParameterMap

用instanceof判断,返回的也是false,有点晕,呵呵

最后,只好用反射来搞定了: 

 

  1. Method method=requestParams.getClass().getMethod( "setLocked" , new  Class[]{ boolean . class });  
  2. method.invoke(requestParams,new  Object[]{ new  Boolean( false )});  

OK!运行通过!锁定已经接触,可以像操作普通的Map一样,修改request的请求参数了!

package com.spzx.common.log.Utils; import jakarta.servlet.http.HttpServletRequest; import jakarta.servlet.http.HttpServletResponse; import lombok.extern.slf4j.Slf4j; import org.springframework.stereotype.Component; import org.springframework.web.method.HandlerMethod; import org.springframework.web.servlet.HandlerInterceptor; import org.springframework.web.servlet.ModelAndView; @Slf4j @Component public class LogInterceptor implements HandlerInterceptor { @Override public boolean preHandle(HttpServletRequest request, HttpServletResponse response, Object handler) throws Exception{ return true; } @Override public void postHandle(HttpServletRequest request, HttpServletResponse response, Object handler, ModelAndView modelAndView) throws Exception{ if (modelAndView!=null){ log.info(“ViewName:”+modelAndView.getViewName()); } } @Override public void afterCompletion(HttpServletRequest request, HttpServletResponse response, Object handler,Exception ex) throws Exception{ HandlerMethod method = (HandlerMethod) handler; ExecuteLog executeLog = method.getMethodAnnotation(ExecuteLog.class); if (executeLog != null && executeLog.isRecord()){ String title = executeLog.logTitle(); LogUtils.saveEcmLog(request,handler,ex,title); } } } package com.spzx.common.log.Utils; import com.alibaba.fastjson2.JSON; import com.spzx.common.log.Dao.SysEcmLogDao; import com.spzx.common.log.entity.SysEcmLog; import jakarta.servlet.http.HttpServletRequest; import org.apache.commons.lang3.StringUtils; import org.springframework.web.method.HandlerMethod; import java.io.PrintWriter; import java.io.StringWriter; import java.lang.reflect.Method; import java.util.Date; import java.util.Map; import java.util.UUID; public class LogUtils { private static SysEcmLogDao ecmLogDao = SpringContextHolder.getBean(SysEcmLogDao.class); public static String getRemoteAddr(HttpServletRequest request){ String remoteAddr = request.getHeader("X-Real-IP"); if (StringUtils.isNotBlank(remoteAddr)){ remoteAddr = request.getHeader("X-Forwarded-For"); }else if (StringUtils.isNotBlank(remoteAddr)){ remoteAddr = request.getHeader("Proxy-Client-IP"); }else if (StringUtils.isNotBlank(remoteAddr)){ remoteAddr = request.getHeader("WL-Proxy-Client-IP"); }else if (StringUtils.isNotBlank(remoteAddr)){ remoteAddr = request.getRemoteAddr(); } return remoteAddr != null ? remoteAddr : request.getRemoteAddr(); } public static String getStackTraceAsString(Throwable e){ if (e == null){ return ""; } StringWriter stringWriter = new StringWriter(); e.printStackTrace(new PrintWriter(stringWriter)); return stringWriter.toString(); } public static void saveEcmLog(HttpServletRequest request,Object handler,Exception ex,String title){ String userId = PrincipalHolder.get(); if (StringUtils.isNotBlank(userId)){ Map paramsMap = request.getParameterMap(); String params = ""; if (paramsMap != null){ params = JSON.toJSONString(request.getParameterMap()); } String remoteAddr = getRemoteAddr(request); String requestUri = request.getRequestURI(); String methodTh = request.getMethod(); SysEcmLog ecmLog = new SysEcmLog(); ecmLog.setId(UUID.randomUUID().toString().replace("-","")); ecmLog.setTitle(title); ecmLog.setRemoteAddr(remoteAddr); ecmLog.setRequestUri(requestUri); ecmLog.setMethod(methodTh); ecmLog.setParams(params); ecmLog.setException(ex == null ? "无" : ex.toString()); ecmLog.setCreateBy(userId); ecmLog.setCreateTime(new Date()); new SaveEcmLogThread(ecmLog,handler,ex); } } public static class SaveEcmLogThread extends Thread{ private SysEcmLog ecmLog; private Object handler; private Exception ex; public SaveEcmLogThread(SysEcmLog ecmLog,Object handler,Exception ex){ this.ecmLog = ecmLog; this.handler = handler; this.ex = ex; } @Override public void run() { if (StringUtils.isBlank(ecmLog.getTitle())){ String permission = ""; if (handler instanceof HandlerMethod){ Method m = ((HandlerMethod) handler).getMethod(); RequestPermissions rp = m.getAnnotation(RequestPermissions.class); permission = (rp!=null ? StringUtils.join(rp.value(),","):""); } } ecmLog.setException(getStackTraceAsString(ex)); if (StringUtils.isBlank(ecmLog.getTitle())&&StringUtils.isBlank(ecmLog.getException())){ return; } ecmLogDao.insert(ecmLog); } } } 报错java,lang.ExceptionInInitializerError:null at com.boe.ecm.customer.config.LogInterceptor.afterCompletion(LogInterceptor.java:52)
最新发布
12-18
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值