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
在 Java Web 开发中,`request.getParameterMap()` 和 `request.getQueryString()` 是处理 HTTP 请求参数的两种方式,它们在使用方式、适用场景以及底层机制上存在显著差异。以下将从多个维度对两者进行详细分析。 ### 参数来源与编码方式 `request.getParameterMap()` 用于获取所有请求参数,并以 `Map<String, String[]>` 的形式返回。该方法能够处理 `GET` 和 `POST` 请求中的参数,并且会自动对参数进行 URL 解码。例如,当请求为 `/user?name=%E5%BC%A0%E4%B8%89` 时,`getParameterMap()` 返回的 `name` 值为 `张三` [^2]。 ```java Map<String, String[]> parameterMap = request.getParameterMap(); ``` 相比之下,`request.getQueryString()` 仅返回 `GET` 请求中的查询字符串部分,即 URL 中 `?` 后面的内容。它返回的是原始的 URL 编码字符串,不会自动解码,需要开发者手动调用 `URLDecoder.decode()` 来解析。 ```java String queryString = request.getQueryString(); ``` ### 数据格式与内容结构 `getParameterMap()` 返回的是结构化的参数集合,适用于参数以 `key=value` 形式传递的场景。例如,`/user?name=zhangsan&age=25` 会被解析为 `{name=[zhangsan], age=[25]}`。如果请求体为 JSON 格式,则 `getParameterMap()` 获取到的 `key` 是整个 JSON 字符串,而 `value` 为空 [^2]。 `getQueryString()` 返回的是原始的字符串形式,适用于需要直接处理 URL 查询参数的场景。例如,`/user?name=zhangsan&age=25` 中的查询字符串为 `name=zhangsan&age=25`,需要开发者自行解析键值对。 ### 使用场景对比 `getParameterMap()` 更适用于传统的表单提交和参数结构清晰的请求。它能够自动处理多种参数格式,包括重复参数(如多选框),并返回数组形式的值。此外,该方法支持从请求体中读取参数,适用于 `POST` 请求。 `getQueryString()` 主要用于处理 `GET` 请求中的参数,适用于需要保留原始 URL 参数格式或进行日志记录的场景。由于其返回的是原始字符串,适用于需要手动解析或转发原始查询字符串的中转逻辑 [^3]。 ### 示例代码 获取并打印 `getParameterMap()` 内容: ```java Map<String, String[]> parameterMap = request.getParameterMap(); for (Map.Entry<String, String[]> entry : parameterMap.entrySet()) { System.out.println(entry.getKey() + ": " + Arrays.toString(entry.getValue())); } ``` 获取并解码 `getQueryString()` 内容: ```java String queryString = request.getQueryString(); if (queryString != null) { try { String decodedQuery = URLDecoder.decode(queryString, "UTF-8"); System.out.println("Decoded Query String: " + decodedQuery); } catch (UnsupportedEncodingException e) { e.printStackTrace(); } } ``` ### 注意事项 - 在使用 `getParameterMap()` 时,若请求内容为 JSON 格式,则无法正确解析键值,仅能获取到整个 JSON 字符串作为键,值为空 。 - `getQueryString()` 仅适用于 `GET` 请求,`POST` 请求中该方法返回 `null`。 - 若需在中转请求时保留原始参数结构,可结合 `getQueryString()` 和 `RequestDispatcher` 实现请求转发 [^3]。
评论
成就一亿技术人!
拼手气红包6.0元
还能输入1000个字符
 
红包 添加红包
表情包 插入表情
 条评论被折叠 查看
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值