博客
关于我
强烈建议你试试无所不能的chatGPT,快点击我
拦截请求返回值的方法
阅读量:7113 次
发布时间:2019-06-28

本文共 3374 字,大约阅读时间需要 11 分钟。

hot3.png

拦截请求返回值的方法 博客分类: java

    像标题所说的,虽然在在JEE开发中并不常见,但也不是没有需要。比如知名的页面装饰框架sitemesh,就是利用filter过滤器先截获返回给客户端的页面,然后分析html代码并最终装饰页面效果后返回给客户端。我这里也先简单的说下如何来自己编程实现,然后说一个我遇到的问题,应该也是大家会遇到但很难找到原因的问题。

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

  1. package
    cc.mzone.test
    ;
  2.  
  3. import
    java.io.CharArrayWriter
    ;
  4. import
    java.io.PrintWriter
    ;
  5.  
  6. import
    javax.servlet.http.HttpServletResponse
    ;
  7. import
    javax.servlet.http.HttpServletResponseWrapper
    ;
  8.  
  9. /**
  10.  * 自定义一个响应结果包装器,将在这里提供一个基于内存的输出器来存储所有
  11.  * 返回给客户端的原始HTML代码。
  12.  *
  13.  * 铁木箱子
  14.  *
  15.  */
  16. public
    class ResponseWrapper
    extends HttpServletResponseWrapper
    {
  17. private
    PrintWriter cachedWriter
    ;
  18. private
    CharArrayWriter bufferedWriter
    ;
  19.  
  20. public ResponseWrapper
    (HttpServletResponse response
    )
    {
  21. super
    (response
    )
    ;
  22. // 这个是我们保存返回结果的地方
  23. bufferedWriter
    =
    new
    CharArrayWriter
    (
    )
    ;
  24. // 这个是包装PrintWriter的,让所有结果通过这个PrintWriter写入到bufferedWriter中
  25. cachedWriter
    =
    new
    PrintWriter
    (bufferedWriter
    )
    ;
  26. }
  27.  
  28. public
    PrintWriter getWriter
    (
    )
    {
  29. return cachedWriter
    ;
  30. }
  31.  
  32. /**
  33. * 获取原始的HTML页面内容。
  34. *
  35. */
  36. public
    String getResult
    (
    )
    {
  37. return bufferedWriter.
    toString
    (
    )
    ;
  38. }
  39. }

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

  1. package
    cc.mzone.test
    ;
  2.  
  3. import
    java.io.IOException
    ;
  4. import
    java.io.PrintWriter
    ;
  5.  
  6. import
    javax.servlet.Filter
    ;
  7. import
    javax.servlet.FilterChain
    ;
  8. import
    javax.servlet.FilterConfig
    ;
  9. import
    javax.servlet.ServletException
    ;
  10. import
    javax.servlet.ServletRequest
    ;
  11. import
    javax.servlet.ServletResponse
    ;
  12. import
    javax.servlet.http.HttpServletResponse
    ;
  13.  
  14. public
    class MyServletFilter
    implements Filter
    {
  15.  
  16. public
    void destroy
    (
    )
    {
  17. // TODO Auto-generated method stub
  18.  
  19. }
  20.  
  21. public
    void doFilter
    (ServletRequest request, ServletResponse response,
  22. FilterChain chain
    )
    throws
    IOException, ServletException
    {
  23. // 使用我们自定义的响应包装器来包装原始的ServletResponse
  24. ResponseWrapper wrapper
    =
    new ResponseWrapper
    (
    (HttpServletResponse
    )response
    )
    ;
  25. // 这句话非常重要,注意看到第二个参数是我们的包装器而不是response
  26. chain.
    doFilter
    (request, wrapper
    )
    ;
  27. // 处理截获的结果并进行处理,比如替换所有的“名称”为“铁木箱子”
  28. String result
    = wrapper.
    getResult
    (
    )
    ;
  29. result
    = result.
    replace
    (
    "名称",
    "铁木箱子"
    )
    ;
  30. // 输出最终的结果
  31. PrintWriter out
    = response.
    getWriter
    (
    )
    ;
  32. out.
    write
    (result
    )
    ;
  33. out.
    flush
    (
    )
    ;
  34. out.
    close
    (
    )
    ;
  35. }
  36.  
  37. @Override
  38. public
    void init
    (FilterConfig filterConfig
    )
    throws ServletException
    {
  39. // TODO Auto-generated method stub
  40.  
  41. }
  42.  
  43. }

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

  1. <filter>
  2. <filter-name>myFilter
    </filter-name>
  3. <filter-class>cc.mzone.test.MyServletFilter
    </filter-class>
  4. </filter>
  5. <filter-mapping>
  6. <filter-name>myFilter
    </filter-name>
  7. <url-pattern>*.jsp
    </url-pattern>
  8. </filter-mapping>

      我们上面配置的是只过滤访问后缀为.jsp的url地址,当然你可以根据自己的需要修改成其他的过滤模式。然后我们在web应用根目录下建立一个jsp文件test.jsp,内容如下:

  1. <%@ page language="java" pageEncoding="UTF-8" contentType="text/html; charset=UTF-8" %>
  2. <html>
  3. <head>
  4. <title>页面返回结果过滤测试
    </title></head>
  5. </head>
  6. <body>
  7. 你好,我叫“名称”。
  8. </body>
  9. </html>

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

  1. 你好,我叫“铁木箱子”

      而不是页面中原始的内容:

  1. 你好,我叫“名称”

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

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

http://www.mzone.cc/article/260.html

http://blog.csdn.net/tangtong1/article/details/53081951

转载于:https://my.oschina.net/xiaominmin/blog/1598675

你可能感兴趣的文章
Photoshop制作一只可爱的卡通小鸟
查看>>
物化视图测试手册
查看>>
Windows7(32bit)用Firefly创建第一个示例
查看>>
《软件性能测试与LoadRunner实战教程》新书上市
查看>>
Windows批处理中的等待技巧
查看>>
艾伟_转载:使用AOP动态调用WebService
查看>>
C#收发邮件(Jmail+Professional+v4.4+特别版)
查看>>
嵌入式计算机与PC机的区别
查看>>
FastReport微调
查看>>
nginx下的文件下载防盗链(HttpAccessKeyModule)
查看>>
Linq延迟执行(转)
查看>>
Django实战(3):Django也可以有scaffold
查看>>
简单缓存 datatable
查看>>
MFC界面的完善
查看>>
WPF&SL之Basic MVVM
查看>>
On :target
查看>>
最新30幅动人心脾的优秀摄影作品欣赏
查看>>
Map 3D 2013 新功能和新API WebCast视频下载
查看>>
2012年7月感想
查看>>
Memcached的分布式算法-Consistent Hashing
查看>>