HttpServletResponse应用(第四次课)
一、简介:
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,找不到客户机所请求的资源)到客户端,并清除缓冲区中的内容。
构建响应消息头:
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方法分别用于得到输出二进制数据、输出文本数据的ServletOuputStream、Printwriter对象。
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 53 response.setContentType("text/html;charset=UTF-8"); //目的是为了控制浏览器的行为,即控制浏览器用UTF-8进行解码; 54 OutputStream out = response.getOutputStream(); 55 String data = "博客"; 56 out.write(data.getBytes("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 //<meta http-equiv="content-type" content="text/html"/> 等价于 response.setContentType("text/html"); 70 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 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>标签模拟响应消息头:
作用等于同上述方法的Refresh和noCache,用与处理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">
详细参考:http://www.cnblogs.com/crazylqy/p/4158026.html
郑重声明:本站内容如果来自互联网及其他传播媒体,其版权均属原媒体及文章作者所有。转载目的在于传递更多信息及用于网络分享,并不代表本站赞同其观点和对其真实性负责,也不构成任何其他建议。