HttpServletRequestWrapper 编码问题

本文介绍了一种解决Spring框架中请求参数出现乱码的方法,通过实现HttpServletRequestWrapper并重写部分方法来确保字符编码正确处理。

摘要生成于 C知道 ,由 DeepSeek-R1 满血版支持, 前往体验 >

当发现经过Wrapper后Spring setCharacterEncoding还乱码时,可以实现HttpServletRequestWrapper,重写public String[] getParameterValues(String name) {....}等方法,再Encoding已设了encoding.

/**
     * 获取应用层的编码方式,应用层Filter需要setCharacterEncoding避免乱码问题
     * 当getCharacterEncoding不为空那么就可以从RequestWrapper中获取转换编码
     * 后的字符串了
     * @param request
     * @return
     */
    public static String getCharacterEncoding(HttpServletRequest request){
        String encoding = request.getCharacterEncoding();
        return encoding != null ? encoding : "ISO8859-1";
    }
    /**
     * 避免Filter乱码问题,默认ISO8859-1,如果应用层setCharacterEncoding需要取出
     * setCharacterEncoding的编码方式
     * @param obj
     * @param encoding
     * @return
     */
    public static String[] encoding(String[] obj,String encoding){
        if (StringUtil.isNotEmpty(obj)) {
            if(!encoding.equalsIgnoreCase("ISO8859-1")){
                try {
                    if (obj.length > 1) {
                        List<String> ls = Arrays.asList((String[]) obj);
                        String[] str = new String[ls.size()];
                        for (int i = 0; i < ls.size(); i++) {
                            str[i] = new String(ls.get(i).getBytes("ISO8859-1"),encoding);
                        }
                        return str;
                    } else {
                        return new String[] {new String(obj[0].getBytes("ISO8859-1"),encoding) };
                    }
                } catch (UnsupportedEncodingException e) {
                    logger.info(e.toString());
                }
            }else{
                return obj;
            }
        }
        return null;
    }


<think>好的,用户的问题是关于如何通过继承HttpServletRequestWrapper来实现多次读取HTTP请求体的示例代码。首先,我需要回忆一下HttpServletRequestWrapper的作用。它通常用于包装HttpServletRequest对象,以便修改或增强其功能。由于HttpServletRequest的输入流(InputStream)只能被读取一次,用户希望多次读取请求体,这就需要缓存请求体的内容。 接下来,我需要考虑如何实现这一点。通常的做法是在包装类中,将输入流的内容读取到一个字节数组或字符串中,并缓存起来。之后,每次获取输入流时,都返回这个缓存的字节数组的输入流。这样就可以多次读取而不会导致流被关闭的问题。 然后,我需要检查是否记得相关的代码结构。比如,自定义的HttpServletRequestWrapper类需要覆盖getInputStream()和getReader()方法,确保它们返回的是基于缓存数据的流。同时,需要在构造方法中读取原始请求的输入流,并将其缓存到字节数组中。 可能需要注意的异常处理,比如IOException的处理,以及确保输入流只能被读取一次,因此缓存操作需要在包装类初始化时完成。此外,还需要考虑字符编码问题,因此在读取字节数组时可能需要使用正确的字符集来转换为字符串。 然后,用户可能需要一个示例代码,所以需要编写一个简单的示例,展示如何继承HttpServletRequestWrapper,缓存body,并提供多次读取的方法。同时,需要说明如何在过滤器或拦截器中使用这个包装类,替换原始的HttpServletRequest,以便后续的处理可以多次读取body。 最后,生成相关问题的时候,需要考虑用户可能进一步想知道的内容,比如缓存body的应用场景、性能影响,或者与Spring框架集成的方法等。确保问题覆盖用户可能遇到的后续疑问。</think>以下是使用`HttpServletRequestWrapper`实现多次读取请求体的Java示例代码: ### 实现步骤 1. **创建自定义包装类**:继承`HttpServletRequestWrapper`并缓存请求体 2. **覆盖输入流方法**:重写`getInputStream()`和`getReader()`方法 3. **缓存处理**:将原始请求的输入流读取到字节数组中 ```java import javax.servlet.ReadListener; import javax.servlet.ServletInputStream; import javax.servlet.http.HttpServletRequest; import javax.servlet.http.HttpServletRequestWrapper; import java.io.BufferedReader; import java.io.ByteArrayInputStream; import java.io.IOException; import java.io.InputStreamReader; public class CachedBodyHttpServletRequest extends HttpServletRequestWrapper { private final byte[] cachedBody; public CachedBodyHttpServletRequest(HttpServletRequest request) throws IOException { super(request); // 读取原始请求体并缓存 this.cachedBody = request.getInputStream().readAllBytes(); } @Override public ServletInputStream getInputStream() { return new CachedBodyServletInputStream(this.cachedBody); } @Override public BufferedReader getReader() { ByteArrayInputStream byteArrayInputStream = new ByteArrayInputStream(this.cachedBody); return new BufferedReader(new InputStreamReader(byteArrayInputStream)); } // 自定义ServletInputStream实现 private static class CachedBodyServletInputStream extends ServletInputStream { private final ByteArrayInputStream buffer; public CachedBodyServletInputStream(byte[] content) { this.buffer = new ByteArrayInputStream(content); } @Override public int read() { return buffer.read(); } @Override public boolean isFinished() { return buffer.available() == 0; } @Override public boolean isReady() { return true; } @Override public void setReadListener(ReadListener listener) { throw new UnsupportedOperationException("Not implemented"); } } } ``` ### 使用示例(在过滤器中) ```java import javax.servlet.*; import javax.servlet.annotation.WebFilter; import javax.servlet.http.HttpServletRequest; import java.io.IOException; @WebFilter("/*") public class CachingFilter implements Filter { @Override public void doFilter(ServletRequest request, ServletResponse response, FilterChain chain) throws IOException, ServletException { HttpServletRequest httpRequest = (HttpServletRequest) request; // 包装原始请求 CachedBodyHttpServletRequest wrappedRequest = new CachedBodyHttpServletRequest(httpRequest); // 后续处理可以使用wrappedRequest多次读取body chain.doFilter(wrappedRequest, response); } } ``` ### 关键点说明 1. **缓存时机**:在构造方法中完成请求体读取,避免后续重复读取原始流[^1] 2. **流重置**:通过`ByteArrayInputStream`实现重复读取 3. **编码处理**:使用`request.getCharacterEncoding()`可保持原始编码设置 4. **线程安全**:每个请求实例独立缓存,适用于并发场景
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值