[size=medium]首先,请看看robbin这文章: http://www.iteye.com/topic/462476
然后,问题:如果请求的网页没有改变,则直接给客户端返回 304 Not Modified 信息,
那这个用java编程怎么实现的呢(怎么修改HTTP响应头信息)?
还有,我感觉tomcat在返回给客户端资源信息时,根本就没有添加 HTTP 响应头信息,
直接给客户端返回HTML内容,为什么呢?是通过response吗?[/size]
一般来说,http的响应头都由tomcat给你处理好了.比如:你请求一个不存在的页面,服务器(tomcat)会返回一个http状态码404
如果请求的网页没有改变,则直接给客户端返回 304 Not Modified 信息,这个如果是静态页面的话(html),服务器也会直接处理了.
对于动态页面(jsp),可以使用response的方法设置响应头.典型的响应头如下:
Date Thu, 11 Nov 2010 14:42:47 GMT
Server Apache/2.0.63 (Unix)
X-Powered-By PHP/5.2.9
X-Pingback http://kenwublog.com/xmlrpc.php
Link http://kenwublog.com/?p=1459; rel=shortlink
x-ua-compatible IE=EmulateIE7
Keep-Alive timeout=15, max=100
Connection Keep-Alive
Transfer-Encoding chunked
Content-Type text/html; charset=UTF-8
给客户端返回 304 Not Modified 信息,是通过设置http状态码来实现的:response.setStatus(304);就可以了.
至于感觉不到响应头,是因为服务器有一些默认的响应头,所以感觉不到
你可以为要请求的页面设置一个过滤器,在里面检查页面内容是否被编辑过(不知道你怎么检查的),如果内容未变,就直接在response里写入响应头信息。好像是用response.setStatus()方法设置状态码的。
可以重新response,去设置信息,返回到客户端。
HttpServletResponseWrapper为我们实现对response对象的后处理提供了帮助——你只需编写一个 HttpServletResponseWrapper的子类,加入自己的功能实现(修饰器模式)。那么子类化 HttpServletResponseWrapper都需要重写那些方法呢?
1、获取response对象,并将输出存放在自定义的流里面,那么关于输出流(outputStream、writer)的操作都是需要重写的了:
1)以流的方式获取输出——重写getOutputStream()
2)以字符方式获取输出——重写getWriter()
3)刷新流——重写flushBuffer()
4)重置流——重写reset()
然后加入新增的获取输出数据的方法就ok了。
2、定义response包装器WapperedResponse继承HttpServletResponseWrapper
[code="java"]public class WapperedResponse extends HttpServletResponseWrapper {
private ByteArrayOutputStream buffer=null;
private ServletOutputStream out=null;
private PrintWriter writer=null;
public GZipResponse(HttpServletResponse resp) throws IOException{
super(resp);
buffer=new ByteArrayOutputStream();//真正存储数据的流
out=new WapperedOutputStream(buffer);
writer=new PrintWriter(new OutputStreamWriter(buffer,this.getCharacterEncoding()));
}
//重载父类获取outputstream的方法
@Override
public ServletOutputStream getOutputStream()throws IOException{
return out;
}
//重载父类获取writer的方法
@Override
public PrintWriter getWriter() throws UnsupportedEncodingException{
return writer;
}
//重载父类获取flushBuffer的方法
@Override
public void flushBuffer()throws IOException{
if(out!=null){
out.flush();
}
if(writer!=null){
writer.flush();
}
}
@Override
public void reset(){
buffer.reset();
}
public byte[] getResponseData()throws IOException{
flushBuffer();//将out、writer中的数据强制输出到WapperedResponse的buffer里面,否则取不到数据
return buffer.toByteArray();
}
//内部类,对ServletOutputStream进行包装
private class WapperedOutputStream extends ServletOutputStream{
private ByteArrayOutputStream bos=null;
public WapperedOutputStream(ByteArrayOutputStream stream) throws IOException{
bos=stream;
}
@Override
public void write(int b) throws IOException{
bos.write(b);
}
}
} [/code]
那么Servlet容器通过调用getOutputStream()方法获得的输出流将是我们自定义的包装流WapperedOutputStream。
3、现在就可以在过滤器中使用WapperedResponse进行对response包装、处理了。
[code="java"]
public void doFilter(ServletRequest arg0, ServletResponse arg1,
FilterChain arg2) throws IOException, ServletException {
HttpServletResponse resp=(HttpServletResponse)arg1;
WapperedResponse wapper=new WapperedResponse(resp);
arg2.doFilter(arg0, wapper);
byte[] b1=wapper.getResponseData();
//do something with b1 here
byte[] b2=...;
//输出处理后的数据
ServletOutputStream output=arg1.getOutputStream();
output.write(b2);
output.flush();
} [/code]