第10章WEB10-requet&response篇
今日任务
- 登录系统后完成文件下载
- 商城系统注册功能.
教学导航
教学目标 |
掌握response设置响应头 掌握response重定向和转发的区别 掌握request接收请求参数 掌握request域的作用范围 |
教学方法 |
案例驱动法 |
1.1 上次课内容回顾:
HTTP协议:
* HTTP协议请求部分:
* 请求行 :请求方式 请求路径 协议版本
* 请求头 :一个key对应一个value ,也有一个key对应多个value
* Referer,User-Agent,If-Modified-Since
* 请求体 :POST方式的参数.
* HTTP协议响应部分:
* 响应行 :协议版本 状态码 状态码描述
* 状态码:200 302 304 404 500
* 响应头 :一个key对应一个value ,也有一个key对应多个value
* Location,Refresh,Last-Modified,禁用本地缓存,Content-Dispostion
* 响应体 :显示到页面的内容.
Servlet:服务器端程序.
* 编写Servlet:
* 编写一个类,实现Servlet接口.继承GenericServlet,HttpServlet.
* 配置Servlet.在web.xml中配置Servlet
* Servlet的执行原理:
* Servlet的生命周期:
* 第一次访问Servlet的时候,服务器创建一个Servlet的对象.init方法就会执行.任何一次请求服务器都会创建一个新的线程执行service方法.service的方法内部根据请求方式调用doXXX方法.当服务器关闭的时候,servlet就会被销毁.destroy方法就会执行.
* Servlet的接口实现关系:
* Servlet的登录的案例:
* 页面提交Servlet
* 获得请求参数:request.getParameter();
* 封装请求参数:
* 调用业务层类:
* 根据处理结果作出响应:
* 页面定时刷新:
* ServletConfig对象:(了解)
* ServletContext对象:(重要)-被多个用户共享.
* 记录网站被登录的次数.
* 在服务器启动的时候ServletContext就会被创建.一个WEB项目创建一个ServletContext对象.
在服务器关闭的时候销毁.域对象-存的数据有作用范围.
* 功能:
* 1.获得文件的MIME的类型.
* 2.获得全局初始化参数.
* 3.存取数据-作为域对象来使用.
* setAttribute(String name,String value);
* getAttribute(String name);
* removeAttribute(String name);
* 4.读取WEB项目的文件:
* getResourceAsStream(String path);
* getRealPath(String path);
1.2 案例一:完成登录用户对他提供文件的下载的功能.
1.2.1 需求:
之前的案例中已经完成的了一个登录的功能,登录失败后回到登录页面.登录成功以后,服务器提供多个文件的下载的链接,点击链接可以完成文件的下载的功能.
1.2.2 分析:
1.2.2.1 技术分析:
【响应对象-Response对象】
- 响应行:
Response设置状态码:
- 响应头:
Response设置响应头:
* 一个key对应一个value
* 一个key对应多个value
- 响应体:
Response设置响应体:
【response处理中文乱码】
- 字节流:
设置浏览器默认打开编码
response.setHeader("Content-Type", "text/html;charset=UTF-8");
中文转成字节数组编码
response.getOutputStream().write("王守义".getBytes("UTF-8"));
- 字符流:
设置response的缓冲区的编码
response.setCharacterEncoding("UTF-8");
设置浏览器默认打开的编码.
response.setHeader("Content-Type", "text/html;charset=UTF-8");
response.setContentType("text/html;charset=UTF-8");相当于上面两句
【文件下载】
一种:超链接方式.(不推荐)
* <a href=”aa.zip”>下载</a>
* <a href=”1.jpg”>下载</a>
二种:手动编码方式完成文件下载.
* 设置两个头和一个流:
* Content-Type:文件MIME的类型.
* Content-Dispostion:
* 文件的输入流:
1.2.2.2 步骤分析:
- 之前的登录案例已经存在:
- 登录成功跳转的页面上提供一组链接:
- 当点击下载的链接之后,提交到Servlet:
- 设置两个头和一个流:
1.2.3 代码实现:
1.在页面中提供一组下载的链接:
response.getWriter().println("<h2>手动编码方式下载</h2>");
response.getWriter().println("<a href='/WEB10/downloadServlet?filename=a.bmp'>a.bmp</a><br/>");
response.getWriter().println("<a href='/WEB10/downloadServlet?filename=WEB01.zip'>WEB01.zip</a>");
2.编写DownloadServlet:
protected void doGet(HttpServletRequest request, HttpServletResponse response) throws ServletException, IOException {
/**
* 接收参数:
* 设置两个头和一个流:
* 文件输入流和响应的输出流对接:
*/
// 接收参数:get乱码 new String("中文".getBytes("ISO-8859-1"),"UTF-8");
String filename = request.getParameter("filename");
System.out.println(filename);
// 设置两个头和一个流:
// 设置Content-Type头
String fileType = this.getServletContext().getMimeType(filename);
response.setContentType(fileType);
// 设置Content-Disposition:
response.setHeader("Content-Disposition", "attachment;filename="+filename);
// 设置文件的输入流:
String path = this.getServletContext().getRealPath("/download/"+filename); //
InputStream is = new FileInputStream(path);
OutputStream os = response.getOutputStream();
int len = 0;
byte[] b = new byte[1024];
while((len = is.read(b))!=-1){
os.write(b, 0, len);
}
is.close();
}
1.2.4 总结:
1.2.4.1 中文文件名下载:
中文文件在不同的浏览器中编码方式不同:
IE :URL编码
Firefox :Base64编码
if(agent.contains("Firefox")){
// 火狐浏览器
filename = base64EncodeFileName(filename);
}else{
// IE,其他浏览器
filename = URLEncoder.encode(filename, "UTF-8");
}
public static String base64EncodeFileName(String fileName) {
BASE64Encoder base64Encoder = new BASE64Encoder();
try {
return "=?UTF-8?B?"
+ new String(base64Encoder.encode(fileName
.getBytes("UTF-8"))) + "?=";
} catch (UnsupportedEncodingException e) {
e.printStackTrace();
throw new RuntimeException(e);
}
}
1.3 案例二:网站的注册的功能的实现:
1.3.1 需求:
在商城的网站上完成一个注册的功能:
1.3.2 分析:
1.3.2.1 技术分析
【请求对象-request对象】
- 获得客户机信息:
* getMethod(); ---获得请求方式.
* getQueryString(); ---获得请求路径后的参数.
* getRemoteAddr(); ---获得客户机的IP地址.
* getRequestURI(); ---获得请求的路径
* getRequestURL(); ---获得请求的路径
* getContextPath(); ---获得工程名
- 获得请求头
* request.getHeader(“User-Agent”);
* request.getHeader(“Referer”);
- 请求参数:
- 用来作为域对象存取值:
1.3.2.2 步骤分析:
- 步骤一:设计注册页面:
- 步骤二:点击页面中的注册的按钮:提交到Servlet中.
- 步骤三:在Servlet中接收参数:
- 步骤四:封装实体中.
- 步骤五:调用业务层.处理数据.
- 步骤六:根据处理的结果做出响应.
1.3.3 代码实现:
1.创建数据库
create database web10;
use web10;
create table user(
id int primary key auto_increment,
username varchar(20),
password varchar(20),
email varchar(20),
name varchar(20),
sex varchar(10),
birthday date,
hobby varchar(50)
);
2.接收参数:
* request.getParameter(String name);
* request.getParameterValues(String name);
* request.getParamaterMap();
3.参数接收中的乱码处理.
POST乱码处理:
* <form method=”post”>的时候,这种提交才是POST提交.其他的都是get.
* POST方式提交的参数在请求体中.request对象在后台接收参数.request对象有缓存区.默认缓冲区的编码ISO-8859-1.
* 处理方式:设置request的缓冲区的编码.
* request.setCharacterEncoding(“UTF-8”);
GET方式乱码处理:
* GET方式提交的参数会在地址栏上显示 在请求行的路径后面.浏览器就会对路径进行一次编码.将编码后内容取出来.再进行一次编码.
* 处理方式:
* 修改服务器提交的编码.
* 采用URLEncoder 和 URLDecoder类对中文进行编码和解码.
* 使用String类的构造方法:
username = new String(username.getBytes("ISO-8859-1"),"UTF-8");
System.out.println(username);
4.数据的封装:BeanUtils.(内省技术)
1.3.4 总结:
1.3.4.1 转发和重定向的区别:
【重定向】
response.sendRedirect(String path); -- 完成重定向
【转发】
request.getRequestDispatcher(String path).forward(request,response);
【区别】
1.转发的地址栏不变的.重定向的地址栏发生变化的.
2.转发是一次请求一次响应,重定向是两次请求两次响应.
3.request域对象存取的值在转发中是有效的,在重定向无效的.
4.转发的路径不需要加工程名.重定向的路径需要加工程名.
1.3.4.2 使用转发和重定向对登录的功能优化:
使用request域保存一个值:在一次请求范围内有效的.
登录失败就可以保存一个错误信息到request中在动态页面中取出这个值${msg}