像标题所说的,虽然在在JEE开发中并不常见,但也不是没有需要。比如知名的页面装饰框架sitemesh,就是利用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;
-
-
/**
-
* 自定义一个响应结果包装器,将在这里提供一个基于内存的输出器来存储所有
-
* 返回给客户端的原始HTML代码。
-
*
-
* @author 铁木箱子
-
*
-
*/
-
public class ResponseWrapper extends HttpServletResponseWrapper {
-
private PrintWriter cachedWriter;
-
private CharArrayWriter bufferedWriter;
-
-
public ResponseWrapper(HttpServletResponse response) {
-
super(response);
-
// 这个是我们保存返回结果的地方
-
bufferedWriter = new CharArrayWriter();
-
// 这个是包装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文件中,如下:
-
<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>
我们上面配置的是只过滤访问后缀为.jsp的url地址,当然你可以根据自己的需要修改成其他的过滤模式。然后我们在web应用根目录下建立一个jsp文件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即可,如下:
-
// 重置响应输出的内容长度
-
response.setContentLength(-1);
-
// 输出最终的结果
-
PrintWriter out = response.getWriter();
-
out.write(result);
-
out.flush();
-
out.close();
http://www.mzone.cc/article/260.html
相关推荐
源代码 博文链接:https://msj.iteye.com/blog/179663
本文将详细介绍如何使用Servlet技术将JSP页面的内容转换为HTML格式,并保存到本地磁盘的过程。 #### 一、理解Servlet与JSP的基本概念 1. **Servlet**: - Servlet是一种Java服务器端应用程序接口(API),它定义了...
2. **Servlet中的中文乱码处理** - **请求乱码**:当用户通过表单提交包含中文的数据时,需要在`doGet`或`doPost`方法中设置请求编码。 ```java request.setCharacterEncoding("UTF-8"); ``` - **响应乱码**:...
javax.servlet.Filter javax.servlet.Servlet javax.servlet.FilterChain javax.servlet.http.Cookie javax.servlet.FilterConfig javax.servlet.ServletConfig javax.servlet.GenericServlet javax.servlet....
Files contained in javax.servlet.jar: META-INF/MANIFEST.MF javax/servlet/http/LocalStrings.properties javax.servlet.http.HttpSessionBindingListener.class javax.servlet....
Called by the servlet container to indicate to a servlet that the servlet is being taken out of service. destroy() - Method in class javax.servlet.GenericServlet Called by the servlet container to ...
`HttpServletResponseWrapper` 是Servlet API中提供的一个抽象类,它允许开发者创建自定义的响应对象,以覆盖或扩展`HttpServletResponse`的行为。这个抽象类提供了一个包装器模式,使得我们可以对响应对象的默认...
4. **内容捕获与保存**:通过`RequestDispatcher`的`include`方法调用指定的JSP页面,并将输出内容写入到内存中的`ByteArrayOutputStream`中。最后将内存中的内容写入到硬盘上的HTML文件中。 #### 关键点总结 - ...
- 在Servlet中,可以使用`response.setContentType("text/html; charset=GBK")`设定响应内容的编码。 - 字符串解码与编码转换:如果接收到的数据是其他编码,如ISO8859-1,需要先解码再编码,例如`new String(str....
在标题“java过滤器中修改一个http请求的返回内容.pdf”和描述中提到的,主要涉及如何在过滤器中通过HttpServletResponseWrapper修改HTTP响应的内容。 首先,我们需要了解HttpServletResponse接口,它是Servlet API...
- 在`getInputStream()`方法中返回该`ServletInputStream`实例。 示例代码: ```java public class RequestWrapper extends HttpServletRequestWrapper { private final byte[] body; public RequestWrapper...
需要注意的是,由于`ServletResponse`是流式的,所以在读取响应内容时可能需要特殊的处理,例如使用Servlet API中的`HttpServletResponseWrapper`。 接下来,我们需要在web.xml或使用Java配置(如果使用Spring框架...
这种方法的优点在于,原有的JSP页面无需改动,只需通过URL调用这个Servlet,动态页面就会被转换为静态页面。例如,访问`http://localhost/test/toHtml?file_name=index`,将会生成一个名为`index.htm`的静态页面。 ...
在Web开发中,为了提高网站的访问速度及减少服务器对数据库的频繁访问,一种常见的做法是将原本动态生成的内容转换为静态页面。本篇文章将深入探讨如何通过JSP(Java Server Pages)技术实现这一目标,并详细分析其...
由于Filter无法直接修改ServletResponse的编码,我们需要在响应内容发送前,通过`HttpServletResponseWrapper`包装原始响应对象,然后在包装类中重写`getOutputStream()`方法,返回一个自定义的`ServletOutputStream...
5. **JSTL(JavaServer Pages Standard Tag Library)**:为了减少在JSP页面中使用脚本,JSTL提供了一套标准标签库,包括核心标签、XML标签、JDBC标签、函数标签等,提高了代码的可读性和可维护性。 6. **EL...
在SCWCD考试中,对Servlet的理解是基础中的基础。 #### javax.servlet - **interface: Servlet**:这是Servlet接口,定义了Servlet生命周期的主要方法,包括`init()`、`service()`和`destroy()`。 - **interface: ...
在现代Web开发中,为了提高页面加载速度,减少网络传输的数据量,开发者通常会使用Gzip工具对JavaScript和CSS文件进行压缩。Gzip是一种广泛使用的数据压缩格式,它能够有效地减小文件大小,从而加快了文件在网络上的...