1. HttpServletResponse
Web服务器收到客户端的http请求,会针对每一次请求,分别创建一个用于代表请求的request对象、和代表响应的response对象。
1.1 响应正文(主体)
getWrite(); 字符输出流 getOutputStream(); 字节输出流
两者不能同时使用
1.2 响应编码
案例:向客户端输出中文
代码语言:javascript复制@Override
protected void doGet(HttpServletRequest req, HttpServletResponse resp)
throws ServletException, IOException {
// 设置响应类型和字符编码为UTF-8, 这样支持汉字显示
// resp.setContentType("text/html");
// resp.setCharacterEncoding("UTF-8");
//设置客户编码类型
resp.setHeader("Content-Type", "text/html;charset=utf-8");
//不设置编译类型,默认编码发送数据 ISO-8859-1(没有中国二字编码),此时会发生乱码
resp.getWriter().write("hello你好");
}
1.3 响应状态码
常见的状态码参照:HTTP状态码详解
- 404请求失败,请求所希望得到的资源未被在服务器上发现
- 200请求已成功,请求所希望的响应头或数据体将随此响应返回。
- 302 重定向
- 500 服务抛出异常
设置响应状态码
1.4 文件下载
代码语言:javascript复制Override
protected void doGet(HttpServletRequest req, HttpServletResponse resp)
throws ServletException, IOException {
String filename = "Java基础考试题卷.docx";
//下载文件
String path = "C:/Users//Desktop/upload/" filename;
FileInputStream fis = new FileInputStream(path);
//设置请求头,文件名需要UTF-8编码
filename = URLEncoder.encode(filename, "UTF-8");
resp.setHeader("Content-disposition", "attachment;filename=" filename);
byte[] bs = new byte[1024];
ServletOutputStream sos = resp.getOutputStream();
int len = 0;
while((len = fis.read(bs)) != -1){
sos.write(bs, 0, len);
}
fis.close();
}
1.5 手写验证码(了解)
实际开发直接调包
输出随机图片(CAPTCHA图像):Completely Automated Public Turing Test to Tell Computers and Humans Apart (全自动区分计算机和人类的测试)
相关主要类(JDK 查看API)
- BufferedImage:内存图像
- Graphics:画笔
- ImageIO:输出图像
Sevlet
代码语言:javascript复制@Override
protected void doGet(HttpServletRequest req, HttpServletResponse resp)
throws ServletException, IOException {
int WIDTH=120;
int HEIGHT=30;
//创建画布
BufferedImage image = new BufferedImage(WIDTH, HEIGHT, BufferedImage.TYPE_INT_RGB);
//创建画笔
Graphics g = image.getGraphics();
//设置画笔颜色
g.setColor(Color.YELLOW);
//画背景
g.fillRect(0, 0, WIDTH, HEIGHT);
//设置干扰线颜色
g.setColor(Color.BLUE);
//加干扰线
for(int i=0;i<6;i ){
int xStart = new Random().nextInt(WIDTH);
int yStart = new Random().nextInt(HEIGHT);
int xEnd = new Random().nextInt(WIDTH);
int yEnd = new Random().nextInt(HEIGHT);
g.drawLine(xStart, yStart, xEnd, yEnd);
}
int x = 5;
//画随机数
for(int i = 0; i< 4;i ){
g.drawString(new Random().nextInt(9) "", x i*30, 20);
}
//输出图片 设置响应格式
resp.setContentType("image/jpeg");
ImageIO.write(image, "JPEG",resp.getOutputStream());
}
html
效果
如何关联静态html和class 1.Html中的src写 /项目名/对应类的@WebServlet注解地址 2.对应类只需要加上路由注解 3.访问的时候为/项目名/html名字.html
1.6 工具类实现验证码
ValidateCode(width,height,码数,干扰数)
1.7 请求重定向
请求重定向指:一个web资源收到客户端请求后,通知客户端去访问另外一个web资源,这称之为请求重定向。
重定向特点:地址栏会变,并发送2次请求,增加服务器负担
实现方式
代码语言:javascript复制response.sendRedirect(request.getContextPath() ”/转发注解”)
实现原理: 302/307状态码和location头即可实现重定向
转发特点:地址栏不会变,客户端发送一次请求,状态码依旧200
代码语言:javascript复制request.getRequestDispatcher(“/转发注解”).forward(request,response)
1.8 其他常用功能
1.8.1 设置缓存禁用
代码语言:javascript复制@Override
protected void doGet(HttpServletRequest req, HttpServletResponse response)
throws ServletException, IOException {
//浏览器刷新时就不会有缓存
response.addHeader("Pragma", "no-cache");
response.setHeader("Cache-Control", "no-cache");
response.setHeader("Expires", "0");
ValidateCode code = new ValidateCode(WIDTH, HEIGHT, 4, 6);
code.write(response.getOutputStream());
}
1.8.2 自动刷新
代码语言:javascript复制protected void doGet(HttpServletRequest request, HttpServletResponse response) throws ServletException, IOException {
// TODO Auto-generated method stub
response.setContentType("text/html;charset=UTF-8");
//response.setHeader("Refresh", "1");//每隔一秒刷新一次
response.setHeader("Refresh", "5;URL=index.html");//3秒后转到另一页面
response.getWriter().write("注册成功!3秒后会自动跳转,苦没有中转点击<a href='index.html'>这里</a>");
}
1.9 Response实现细节
- getOutputStream和getWriter方法分别用于得到输出二进制数据、输出文本数据
- getOutputStream和getWriter这两个方法互相排斥,调用了其中的任何一个方法后,就不能再调用另一方法。 会抛异常。
- Servlet程序向ServletOutputStream或PrintWriter对象中写入的数据将被Servlet引擎从response里面获取,Servlet引擎将这些数据当作响应消息的正文,然后再与响应状态行和各响应头组合后输出到客户端。
- Serlvet的service方法结束后,Servlet引擎将检查getWriter或getOutputStream方法返回的输出流对象是否已经调用过close方法,如果没有,Servlet引擎将调用close方法关闭该输出流对象。
2. HttpServletRequest
HttpServletRequest对象代表客户端的请求,当客户端通过HTTP协议访问服务器时,HTTP请求头中的所有信息都封装在这个对象中,开发人员通过这个对象的方法,可以获得客户这些信息。
2.1 常用方法
2.1.1 获取客户端信息
- getRequestURL方法返回客户端发出请求时的完整URL。
- getRequestURI方法返回请求行中的资源名部分。
- getQueryString 方法返回请求行中的参数部分。
- getRemoteAddr方法返回发出请求的客户机的IP地址
- getRemoteHost方法返回发出请求的客户机的完整主机名
- getRemotePort方法返回客户机所使用的网络端口号
- getLocalAddr方法返回WEB服务器的IP地址。
- getLocalName方法返回WEB服务器的主机名
- getMethod得到客户机请求方式
2.1.2 获取请求头
- getHead(name)方法 通过key的name获得value
- getHeaders(String name)方法 获得所有key
- getHeaderNames方法
2.1.3 获取请求参数(核心)
- getParameter(name)获取一个参数的一个值
- getParameterValues(String name)获取一个参数的多个值
- getParameterNames方法 获取所有参数的名字
- getParameterMap方法 将参数:值[] 的形式存在map中
- getInputStream
Map查询效率非常高,通常用上图方法来遍历Map(entrySet返回Entry,然后通过entry查询
2.2 应用
2.2.1 获取表单数据一 getParameter
代码语言:javascript复制目标:掌握通过getParameter和getParameterValues获取请求参数 掌握POST和GET请求解决乱码的方案 掌握表单提交通过都用POST方法
<form action="/项目名/注解内的地址" method="post">
用户名:<input type="text" name="username"><br>
密---码:<input type="password" name="password"><br>
性---别:<input type="radio" name="gender" value="男">男
<input type="radio" name="gender" value="女">女<br>
爱---好:<input type="checkbox" name="hobby" value="写代码">写代码
<input type="checkbox" name="hobby" value="看书">看书
<input type="checkbox" name="hobby" value="听课">听课<br>
备---注:<textarea rows="5" cols="20" name="remark"></textarea><br>
<input type="submit" value="提交">
</form>
代码语言:javascript复制protected void doGet(HttpServletRequest request, HttpServletResponse response) throws ServletException, IOException {
//接收表单请求参数
request.setCharacterEncoding("UTF-8");
String username = request.getParameter("username");
String password = request.getParameter("password");
String gender = request.getParameter("gender");
String remark = request.getParameter("remark");
String[] hobby = request.getParameterValues("hobby");
System.out.println(username);
System.out.println(password);
System.out.println(gender);
System.out.println(remark);
System.out.println(Arrays.toString(hobby));
}
2.2.2 获取表单数据二 getParameterMap 存入model
Model
Servlet
代码语言:javascript复制public class Test extends HttpServlet{
@Override
protected void doPost(HttpServletRequest request, HttpServletResponse resp) throws ServletException, IOException {
//post请求时,中文会乱码,解决方法setCharacterEncoding,只针对post请求有效
request.setCharacterEncoding("UTF-8");
//创建用户对象
User user = new User();
//获取请求参数
/**
* key value
* username {"shunxu"};
* password {"123"};
* hobby {"study","reading"}
*/
Map<String, String[]> map = request.getParameterMap();
try {
for(Entry<String, String[]> entry : map.entrySet()){
String pname = entry.getKey();
//通过name创建对象
if(pname.equals("username")){
user.setUsername(entry.getValue()[0]);
}else if(pname.equals("hobby")){
user.setHobby(entry.getValue());
}
}
} catch (Exception e) {
// TODO Auto-generated catch block
e.printStackTrace();
}
System.out.println(user);
}
}
2.2.3 获取表单数据三 通过beanutils封装参数(开发推荐)
导入
- commons-beanutils
- commons-logging
如果不知道如何选择版本,比较靠谱的方式是去Maven Repository搜索包名,查看最多人用的版本
调包使用
BeanUtils自动将request.getParameterMap()中获取到的Map装载入user中。
2.2.4 获取原始表单数据 request.getInputStream
2.2.5 实现请求转发
请求转发指一个web资源收到客户端请求后,通知服务器去调用另外一个web资源进行处理。 request对象提供了一个getRequestDispatcher方法,该方法返回一个RequestDispatcher对象,调用这个对象的forward方法可以实现请求转发。 request对象同时也是一个域对象(context对象),开发人员通过request对象在实现转发时,把数据通过request对象带给其它web资源处理。
- 在作用域中存取数据
- setAttribute方法
- getAttribute方法
- removeAttribute方法
- getAttributeNames方法
2.2.6 实现请求重定向
重定向机制的运作流程
- 1.用户在浏览器端输入特定URL,请求访问服务器端的某个组件
- 2.服务器端的组件返回一个状态码为302的响应结果。
- 3.当浏览器端接收到这种响应结果后,再立即自动请求访问另一个web组件
- 4.浏览器端接收到来自另一个web组件的响应结果。
HttpServeltResponse的sendRedirect(String location)用于重定向
代码语言:javascript复制@WebServlet("/test")
public class Test extends httpServlet{
private static final long serialVersionUID =1L;
protected void doGet(HttpServletRequest request,HttpServeltResponse response)
throws Exception{
String cp = request.getContextPath();
System.out.println(cp); //打印上下文路径
response.sendRedirect(cp "/login.jsp"); //跳转到jsp页面,需要加上contextPath
}
2.2.7 转发和重定向的区别
域对象:session,request,page等(详情关注后续章节) 域:区间、范围
- 重定向:以前的request中存放的变量全部失效!! 并进入一个新的request作用域。
- 转发:以前的request中存放的变量不会失效,就像把两个页面拼到了一起。
转发存放在Request中的变量不会失效
2.2.8 include()、forward()、sendRedirect()的区别(重要)
顾名思义先翻译:
- include 包含:可以理解为将两个页面包含在一起
- forward 向前(传递):理解为将“内容完全托付给下一个”request
- sendRedirect 发送重定向:重新确定方向,重新定位servlet
假定第一次请求的为servlet1,处理转发的为servlet2
include()方法将请求转发给servlet2,servle2对该请求做出了的响应并入到原来的servlet1响应对象中,原来的servlet1还可以继续输出响应信息(servlet1和servlet2按顺序依次响应数据,就像是一个页面一样)。
forward方法将请求转发给其他的servlet2,servlet2负责对请求做出响应,而原先的servlet1的执行则终止(servlet1无法响应任何数据)。
sendRedict()在浏览器请求servlet1之后,重新告诉浏览器将请求重新定位到servlet。
具体结果读者自行尝试