servlet中使用HttpServletResponseWrapper截获返回的页面内容

本文介绍了一种使用Java Servlet过滤器实现的页面内容修改方法。通过自定义`ResponseWrapper`类和`MyServletFilter`过滤器,可以在不改动原始业务代码的情况下,对返回给客户端的HTML内容进行动态替换。

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

思路:就是利用filter过滤器先截获返回给客户端的页面,然后分析html代码并最终装饰页面效果后返回给客户端。

 要截获页面返回的内容,整体的思路是先把原始返回的页面内容写入到一个字符Writer,然后再组装成字符串并进行分析,最后再返回给客户端。

代码如下:


package cc.mzone.test;

import java.io.CharArrayWriter;
import java.io.PrintWriter;

import javax.servlet.http.HttpServletResponse;
import javax.servlet.http.HttpServletResponseWrapper;

public class ResponseWrapper extends HttpServletResponseWrapper {

    private PrintWriter cachedWriter;

    private CharArrayWriter bufferedWriter;

    public ResponseWrapper(HttpServletResponse response) {

        super(response);

        // 这个是我们保存返回结果的地方

        bufferedWriter = new CharArrayWriter();
        
        /*CharArrayWriter 实现了以数组作为目标的输出流。CharArrayWriter 有两个构造函数:

        CharArrayWriter( )

        CharArrayWriter(int numChars)

        第一种形式,创建了一个默认长度的缓冲器。
        第二种形式,缓冲器长度由numChars指定。缓冲器保存在CharArrayWriter的buf 成员中。
        缓冲器大小在需要的情况下可以自动增长。
        缓冲器保持的字符数包含在CharArrayWriter的count 成员中。
        buf 和count 都是受保护的域。

        // 这个是包装PrintWriter的,让所有结果通过这个PrintWriter写入到bufferedWriter中
       */
        cachedWriter = new PrintWriter(bufferedWriter);

    }

    @Override
    public PrintWriter getWriter() {

        return cachedWriter;

    }

    /**
     *
     * 获取原始的HTML页面内容。
     *
     * @return
     */

    public String getResult() {

        return bufferedWriter.toString();

    }

}



//然后再写一个过滤器来截获内容并处理:


package cc.mzone.test;

import java.io.IOException;
import java.io.PrintWriter;

import javax.servlet.Filter;
import javax.servlet.FilterChain;
import javax.servlet.FilterConfig;
import javax.servlet.ServletException;
import javax.servlet.ServletRequest;
import javax.servlet.ServletResponse;
import javax.servlet.http.HttpServletResponse;

public class MyServletFilter implements Filter {

    @Override
    public void destroy() {

        // TODO Auto-generated method stub

    }

    @Override
    public void doFilter(ServletRequest request, ServletResponse response,
            FilterChain chain) throws IOException, ServletException {

        // 使用我们自定义的响应包装器来包装原始的ServletResponse

        ResponseWrapper wrapper = new ResponseWrapper(
                (HttpServletResponse) response);

        // 这句话非常重要,注意看到第二个参数是我们的包装器而不是response

        chain.doFilter(request, wrapper);

        // 处理截获的结果并进行处理,比如替换所有的“名称”为“铁木箱子”

        String result = wrapper.getResult();

        result = result.replace("名称", "铁木箱子");

        // 输出最终的结果

        PrintWriter out = response.getWriter();

        out.write(result);

        out.flush();

        out.close();

    }

    @Override
    public void init(FilterConfig filterConfig) throws ServletException {

        // TODO Auto-generated method stub

    }

}

//然后将该servlet配置在web.xml文件中,如下:


<?xml version="1.0" encoding="UTF-8"?>
<web-app xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"
    xmlns="http://java.sun.com/xml/ns/javaee"
    xsi:schemaLocation="http://java.sun.com/xml/ns/javaee http://java.sun.com/xml/ns/javaee/web-app_3_0.xsd"
    id="WebApp_ID" version="3.0">
    <display-name>Test</display-name>

    <filter>

        <filter-name>myFilter</filter-name>

        <filter-class>cc.mzone.test.MyServletFilter</filter-class>

    </filter>

    <filter-mapping>

        <filter-name>myFilter</filter-name>

        <url-pattern>*.jsp</url-pattern>

    </filter-mapping>

</web-app>


test.jsp页面内容如下:

<%@ page language="java" pageEncoding="UTF-8" contentType="text/html; charset=UTF-8" %>

<html>
<head>
    <title>页面返回结果过滤测试</title></head>
</head>
<body>
你好,我叫“名称”。
</body>
</html>

//  配置完后,部署到tomcat,然后访问应用下的test.jsp文件,就可以发现返回的内容变成了:

你好,我叫“铁木箱子”
补充:

从而也就达到了我们想要的效果了。在文章开头我也提到了说有一个问题,那就是有可能在运行的过程中页面只输出一部分,尤其是在使用多个框架后(比如sitemesh)出现的可能性非常大,在探究了好久之后终于发现原来是响应的ContentLength惹的祸。因为在经过多个过滤器或是框架处理后,很有可能在其他框架中设置了响应的输出内容的长度,导致浏览器只根据得到的长度头来显示部分内容。知道了原因,处理起来就比较方便了,我们在处理结果输出前重置一下ContentLength即可,如下:


  1. // 重置响应输出的内容长度
  2. response.setContentLength(-1);
  3. // 输出最终的结果
  4. PrintWriter out = response.getWriter();
  5. out.write(result);
  6. out.flush();
  7. out.close();


评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值