一、简介:

 

    Web服务器收到客户端的http请求,会针对每一次请求,分别创建一个用于代表请求的request对象、和代表响应的response对象。

request和response对象即然代表请求和响应,那我们要获取客户机提交过来的数据,只需要找request对象就行了。要向客户机输出数据,只需要找response对象就行了。

WEB服务器回送给WEB客户端的HTTP响应消息分为三个部分:

状态行

 

二、响应消息头

 

消息正文(也叫实体内容)

Servlet API中定义的ServletResponse接口类用于创建响应消息。

HttpServletResponse是专用于HTTP协议的ServletResponse子接口,它用于封装HTTP响应消息。

HttpServletResponse定义了一系列用于描述各种HTTP状态码的常量。

在service()方法内部调用HttpServletResponse对象的各种方法来创建响应消息。

 

产生响应状态行:

HTTP响应消息的响应状态行包括HTTP版本、状态代码和一条相关的提示信息:

    HTTP/1.1 200 OK

HttpServletResponse中定义了若干与状态码数值对应的常量,每个常量的名称以前缀SC(Status Code的简写)开头,然后是状态码在HTTP 1.1规范中所表示的状态信息的英文单词的组合,每个单词之间用下划线连接,且所有字母都大写。

    状态码404 对应的常量为HttpServletResponse.SC_NOT_FOUND 

setStatus方法用于设置HTTP响应消息的状态码,并生成响应状态行。

sendError方法用于发送表示错误信息的状态码(一般是404,找不到客户机所请求的资源)到客户端,并清除缓冲区中的内容。



response.sendError(HttpServletResponse.SC_NOT_FOUND);



 

 

构建响应消息头:

addHeader与setHeader方法

addIntHeader与setIntHeader方法

addDateHeader与setDateHeader方法

setContentLength方法

setContentType方法:用于设置Servlet输出内容的MIME类型,对于HTTP协议来说,就是设置Content-Type响应头字段的值。如“text/html;charset=UTF-8”

setCharacterEncoding方法:用于设置输出内容的MIME声明中的字符集编码,对HTTP协议来说,就是设置Content-Type头字段中的字符编码部分。

 

三、response细节:

getOutputStream和getWriter方法分别用于得到输出二进制数据、输出文本数据的ServletOutputStream、PrintWriter对象。



HttpServletResponse response=(HttpServletResponse)arg1;
        ServletOutputStream so=response.getOutputStream();
        PrintWriter printwriter=response.getWriter();



 

getOutputStream和getWriter这两个方法互相排斥,调用了其中的任何一个方法后,就不能再调用另一方法。 

 

Servlet程序向ServletOutputStream或PrintWriter对象中写入的数据将被Servlet引擎从response里面获取,Servlet引擎将这些数据当作响应消息的正文,然后再与响应状态行和各响应头组合后输出到客户端。

 

Serlvet的service方法结束后,Servlet引擎将检查getWriter或getOutputStream方法返回的输出流对象是否已经调用过close方法,如果没有,Servlet引擎将调用close方法关闭该输出流对象。

 

 

例子1

使用了springMvc

 



1 <%@ page language="java" contentType="text/html; charset=utf-8"
 2     pageEncoding="utf-8"%>
 3 <!DOCTYPE html PUBLIC "-//W3C//DTD HTML 4.01 Transitional//EN" "http://www.w3.org/TR/html4/loose.dtd">
 4 <html>
 5 <head>
 6 <meta http-equiv="Content-Type" content="text/html; charset=utf-8">
 7 <title>Insert title here</title>
 8 </head>
 9 <body>
10 <div><a href="responseDownload">responseDownload</a></div>
11 <div><a href="responseLuanMa1">responseLuanMa1</a></div>
12 <div><a href="responseLuanMa2">responseLuanMa2</a></div>
13 <div><a href="Refresh">Refresh</a></div>
14 <div><a href="noCache">noCache</a></div>
15 </body>
16 </html>



 

处理程序



1 package controller;
 2 import java.io.*;
 3 import java.net.URLEncoder;
 4 import javax.servlet.ServletContext;
 5 import javax.servlet.http.HttpServletRequest;
 6 import javax.servlet.http.HttpServletResponse;
 7 import org.springframework.stereotype.Controller;
 8 import org.springframework.web.bind.annotation.RequestMapping;
 9 import org.springframework.web.servlet.ModelAndView;
10 import bean.LoginForm;
11  
12  
13 @Controller
14 public class ResponseController {
15     @RequestMapping(value="responseIndex")
16     public ModelAndView responseIndex(HttpServletRequest request,HttpServletResponse response){      
17         ModelAndView mv = new ModelAndView("/Response/responseIndex");
18         return mv;
19     }
20     
21     @RequestMapping(value="responseMethod")
22     public void responseMethod(HttpServletRequest request,HttpServletResponse response){      
23         
24     }
25 //    实现文件的下载
26 //    需要通过HttpServletResponse.setHeader方法设置Content-Disposition头的值为“attachment; filename =文件名”。
27 //    应该调用HttpServletResponse.getOutputStream方法返回的ServletOutputStream对象来向客户 端写入附件文件内容,而不应使用HttpServletResponse.getWriter方法返回的PrintWriter对象。   
28     @RequestMapping(value="responseDownload")
29     public void responseDownload(HttpServletRequest request,HttpServletResponse response) throws Exception{  
30         ServletContext context=request.getSession().getServletContext();
31         String Stringpath = context.getRealPath("/WEB-INF/lqy.txt");  
32         String filename = Stringpath.substring(Stringpath.lastIndexOf("\\")+1);
33         //2.通知浏览器以下载方式打开发送过来的数据  
34         //如果文件名是中文,要经过URL编码    
35         //下面这句就是以下载方式打开,没有的话就会直接把p.txt的内容直接打印出来
36         response.setHeader("content-disposition","attachment;filename="+URLEncoder.encode(filename,"UTF-8")); 
37         
38         //3.读取资源内容  
39         FileInputStream fis = new FileInputStream(Stringpath);
40         byte[]buffer = new byte[1024];
41         int len = 0;  
42         while((len=fis.read(buffer))>0){  
43                 //关键语句
44                response.getOutputStream().write(buffer,0,len);  
45         }  
46         fis.close();              
47     }
48     
49     //解决办法之一
50     @RequestMapping(value="responseLuanMa1")
51     public void responseLuanMa1(HttpServletRequest request,HttpServletResponse response) throws Exception{ 
52         //<meta http-equiv="Content-Type"content="text/html; charset=UTF-8"> 等价于 response.setContentType("text/html;charset=UTF-8"); 
53         response.setContentType("text/html;charset=UTF-8"); //目的是为了控制浏览器的行为,即控制浏览器用UTF-8进行解码;和html中的meta中设置Content-Type的chart-set一样原理
54         OutputStream out = response.getOutputStream();  
55         String data = "博客";  
56         out.write(data.getBytes("UTF-8"));//输出程序未utf-8编码
57         //输出乱码的问题是程序用UTF-8编码,而浏览器用GB2312解码,因此会出现乱码;这样处理就把浏览器和程序的编码一样了      
58     }
59     //解决办法之二(处理PrintWriter)
60     @RequestMapping(value="responseLuanMa2")
61     public void responseLuanMa2(HttpServletRequest request,HttpServletResponse response) throws Exception{ 
62         //response.setCharacterEncoding("GB2312"); //因为IE和WEB服务器之间不能传输文本,然后就通过ISO-8859-1进行编码,但是ISO-8859-1中没有“博客”的编码,因此输出“??”表示没有编码;
63         response.setCharacterEncoding("UTF-8");        //GB2312是针对IE,假如其他浏览器呢?改成UTF-8适应其他浏览器
64         PrintWriter out = response.getWriter();  
65         String data = "博客";  
66         out.println(data);       
67     }
68     
69  
70     //意思是设置meta,详细属性可以查看meta
71     @RequestMapping(value="Refresh")
72     public void Refresh(HttpServletRequest request,HttpServletResponse response) throws Exception{ 
73         //这个是不停的刷新
74         //response.setHeader("Refresh","2");
75         //在Refresh头字段的时间设置值后面还可以用分号(;)分隔后,再指定一个URL地址,这将让浏览器在指定的时间值后自动去访问该URL地址指向的资源。
76         response.setHeader("Refresh","2;responseDownload");
77     }
78     //意思是设置meta,详细属性可以查看meta
79     @RequestMapping(value="noCache")
80     public void noCache(HttpServletRequest request,HttpServletResponse response) throws Exception{ 
81         response.setDateHeader("Expires",0 );
82         response.setHeader("Cache-Controll","no-cache");
83         response.setHeader("pragma","no-cache");
84     }   
85     
86     
87     
88     
89 }



 


 

扩展-使用<meta>标签模拟响应消息头:

 

作用等于同上述方法的

 



response.setContentType("text/html;charset=UTF-8");



response.setDateHeader("Expires",0 );
response.setHeader("Cache-Controll","no-cache");
response.setHeader("pragma","no-cache");

 

用与处理WEB服务器发送的响应消息头一样的方式来进行处理。

 

HTML语言中专门定义了<meta>标签的http-equiv属性来在HTML文档中模拟HTTP响应消息头,当浏览器读取到 HTML文档中具有http-equiv属性的<meta>标签时,它会用与处理WEB服务器发送的响应消息头一样的方式来进行处理。 

举例:

<meta http-equiv="Expires" content="0">
<meta http-equiv="Cache-Control"content="no-cache">
<meta http-equiv="Pragma"content="no-cache">
<meta http-equiv="Refresh"content="0;url=http://www.it315.org">
<meta http-equiv="Content-Type"content="text/html; charset=GB2312">