代码1:在不跳转下访问目标jsp。就是利用RequestDispatcher.include(ServletRequest request, ServletResponse response)。该方法把RequestDispatcher指向的目标页面写到response中。
代码1- public static String getJspOutput(String jsppath, HttpServletRequest request, HttpServletResponse response)
 - throws Exception
 - {
 - WrapperResponse wrapperResponse = new WrapperResponse(response);
 - request.getRequestDispatcher(jsppath).include(request, wrapperResponse);
 - return wrapperResponse.getContent();
 -  }   
 
此处一个自定义类WrapperResponse封装了HttpServletResponse,具体请看一下代码
代码2:HttpServletResponse的封装类,继承自HttpServletResponseWrapper。其核心是构建一个OutputStream,且让HttpServletResponse获得的jsp内容输入到这个OutputStream。
java 代码2- import java.io.ByteArrayOutputStream;
 - import java.io.IOException;
 - import java.io.PrintWriter;
 - import java.io.UnsupportedEncodingException;
 - import javax.servlet.http.HttpServletResponse;
 - import javax.servlet.http.HttpServletResponseWrapper;
 - public class WrapperResponse extends HttpServletResponseWrapper {
 - private MyPrintWriter tmpWriter;
 - private ByteArrayOutputStream output;
 - public WrapperResponse(HttpServletResponse httpServletResponse) {
 - super(httpServletResponse);
 - output = new ByteArrayOutputStream();
 - tmpWriter = new MyPrintWriter(output);
 - }
 - public void finalize() throws Throwable {
 - super.finalize();
 - output.close();
 - tmpWriter.close();
 - }
 - public String getContent() {
 - try {
 - tmpWriter.flush(); //刷新该流的缓冲,详看java.io.Writer.flush()
 - String s = tmpWriter.getByteArrayOutputStream().toString("UTF-8");
 - //此处可根据需要进行对输出流以及Writer的重置操作
 - //比如tmpWriter.getByteArrayOutputStream().reset()
 - return s;
 - } catch (UnsupportedEncodingException e) {
 - return "UnsupportedEncoding";
 - }
 - }
 - //覆盖getWriter()方法,使用我们自己定义的Writer
 - public PrintWriter getWriter() throws IOException {
 - return tmpWriter;
 - }
 - public void close() throws IOException {
 - tmpWriter.close();
 - }
 - //自定义PrintWriter,为的是把response流写到自己指定的输入流当中
 - //而非默认的ServletOutputStream
 - private static class MyPrintWriter extends PrintWriter {
 - ByteArrayOutputStream myOutput; //此即为存放response输入流的对象
 - public MyPrintWriter(ByteArrayOutputStream output) {
 - super(output);
 - myOutput = output;
 - }
 - public ByteArrayOutputStream getByteArrayOutputStream() {
 - return myOutput;
 - }
 - }
 -  }    
 
转引结束
利用上面的WrapperResponse 类,写了一个filter,实现在jsp展现之前,获取即将展现的jsp的全部内容
获取这个内容的好处不多说(如果不知道就不用看下去了)
下面的代码利用filter,实现对字符串""的过滤,并转换成"邪教"显示
java 代码- import java.io.IOException;
 - 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.HttpServletRequest;
 - import javax.servlet.http.HttpServletResponse;
 - public class GavinFilter implements Filter
 - {
 - public void destroy()
 - {
 - // TODO 自动生成方法存根
 - }
 - public void doFilter(ServletRequest arg0, ServletResponse arg1, FilterChain chain) throws IOException, ServletException
 - {
 - HttpServletRequest request = (HttpServletRequest)arg0;
 - HttpServletResponse response = (HttpServletResponse)arg1;
 - WrapperResponse wrapperResponse = new WrapperResponse(response);
 - chain.doFilter(request, wrapperResponse);
 - String html = wrapperResponse.getContent();
 - response.getWriter().print(html.replaceAll("", "邪教"));
 - }
 - public void init(FilterConfig arg0) throws ServletException
 - {
 - // TODO 自动生成方法存根
 - }
 - public static String getJspOutput(String jsppath, HttpServletRequest request, HttpServletResponse response) throws Exception
 - {
 - WrapperResponse wrapperResponse = new WrapperResponse(response);
 - request.getRequestDispatcher(jsppath).include(request, wrapperResponse);
 - return wrapperResponse.getContent();
 - }
 -  }   
 
Web服务器接收到客户端的http请求,会针对每一次请求,分别创建一个用于代表请求的request对象、和代表响应的response对象。
 request和response对象既然代表请求和响应,那么我们要获取客户机提交过来的数据,只需要找request对象,就行了。要向客户机输出数据,只需要找response对象就行了。
 
 HttpServletRequest封装客户端相关信息,服务器Servlet程序可以通过request对象操作客户端信息
 HttpServletResponse封装服务器向客户端发送响应数据信息,Servlet程序通过response对象向客户端发送响应
response
 1.response常用API
 setStatus:设置响应行当中的状态码
 setHeader:设置响应头信息
 getOutputStream:获得字节流 --- 输出响应体内容
 getWriter:获得字符流 --- 输出响应体内容
 2.HttpServletResponse继承ServletResponse接口,ServletResponse并没有提供与HTTP协议相关API,HttpServletResponse添加了与协议相关API
 JavaEE API 中并没有提供HttpServletResponse实现类---实现类由tomcat服务器提供的
 3.常用状态码:200 302 304 404 500
 200 请求处理成功
 302 客户端重定向
 304 客户端访问资源没有被修改,客户端访问本地缓存
 404 访问资源不存在
 500 服务器内部出错
 
因为WrapperResponse extends HttpServletResponseWrapper extends ServletResponseWrapper,类ServletResponseWrapper的getWriter()方法实现:
 
    public PrintWriter getWriter() throws IOException {
         return this.response.getWriter();
     }
 
如果一个类继承了父类,并且重写这个方法,用于自己的逻辑。
 
 
 
- 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文件中,如下:
- >
 - >myFilter >
 - >cc.mzone.test.MyServletFilter >
 - >
 - >
 - >myFilter >
 - >*.jsp >
 - >
 
我们上面配置的是只过滤访问后缀为.jsp的url地址,当然你可以根据自己的需要修改成其他的过滤模式。然后我们在web应用根目录下建立一个jsp文件test.jsp,内容如下:
- <%@ page language="java" pageEncoding="UTF-8" contentType="text/html; charset=UTF-8" %>
 - >
 - >
 - <span style="color: rgb(0, 0, 0); font-weight: bold;">></span></span></span>页面返回结果过滤测试<span style="color: rgb(0, 153, 0);"><span style="color: rgb(0, 0, 0); font-weight: bold;">> >
 - >
 - >
 - 你好,我叫“名称”。
 - >
 -  
      
      > 
      
配置完后,部署到tomcat,然后访问应用下的test.jsp文件,就可以发现返回的内容变成了:
- 你好,我叫“铁木箱子”
 
而不是页面中原始的内容:
- 你好,我叫“名称”
 
从而也就达到了我们想要的效果了。在文章开头我也提到了说有一个问题,那就是有可能在运行的过程中页面只输出一部分,尤其是在使用多个框架后(比如sitemesh)出现的可能性非常大,在探究了好久之后终于发现原来是响应的ContentLength惹的祸。因为在经过多个过滤器或是框架处理后,很有可能在其他框架中设置了响应的输出内容的长度,导致浏览器只根据得到的长度头来显示部分内容。知道了原因,处理起来就比较方便了,我们在处理结果输出前重置一下ContentLength即可,如下:
- // 重置响应输出的内容长度
 - response. setContentLength ( - 1 ) ;
 - // 输出最终的结果
 - PrintWriter out = response. getWriter ( ) ;
 - out. write (result ) ;
 - out. flush ( ) ;
 - out. close ( ) ;
 
这样处理后就不会再出现只出现部分页面的问题了!
 
