JavaWeb - Response笔记

主要内容

  1. HTTP协议:响应消息
  2. Response对象
  3. ServletContext对象

一、HTTP协议:

1-请求消息:客户端发送给服务器端的数据

数据格式:请求行、请求头、请求空行、请求体;

2-响应消息:服务器端发送给客户端的数据

数据格式:响应行、头、空行、体

1-响应行

  1. 组成:协议/版本 响应状态码 状态码描述
  2. 响应状态码:服务器告诉客户端浏览器本次请求和响应的一个状态。
    1. 状态码都是3位数字
    2. 分类:

      1xx:服务器接收客户端消息,但没有接收完成,等待一段时间后,发送1xx多状态码
      2xx:成功。代表:200
      3xx:重定向。代表:302(重定向),304(访问缓存)
      4xx:客户端错误。
      代表:
      404:请求路径没有对应的资源)
      405:请求方式没有对应的doXxx方法
      5xx:服务器端错误。代表:500(服务器内部出现异常)

2-响应头

  1. 格式:头名称: 值
    2. 常见的响应头:

     > Content-Type:服务器告诉客户端本次**<u>响应体数据格式以及编码格式</u>**
     >
     > Content-disposition:服务器告诉客户端**以什么格式打开响应体数据**
     > 	值:
     > 		in-line:默认值,在当前页面内打开
     > 		attachment;filename=xxx:以附件形式打开响应体。文件下载
    

3-响应空行

4-响应体:传输的数据

<!-- 响应字符串格式 -->
HTTP/1.1 200 OK
Content-Type: text/html;charset=UTF-8
Content-Length: 101
Date: Wed, 06 Jun 2018 07:08:42 GMT
<html>
  <head>
    <title>$Title$</title>
  </head>
  <body>
  hello , response
  </body>
</html>

二、Response对象

功能:设置响应消息

设置响应行

  1. 格式:HTTP/1.1 200 ok
  2. 设置状态码:setStatus(int sc)

设置响应头:

​ setHeader(String name, String value)

设置响应体:

  • 使用步骤:
    1. 获取输出流:

字符输出流:PrintWriter getWriter()
字节输出流:ServletOutputStream getOutputStream()

2. 使用输出流,将数据输出到客户端浏览器。

案例:

案例一:完成重定向

  • 重定向:资源跳转的方式

  • 代码实现:

    //访问/responseDemo1,会自动跳转到/responseDemo2资源
    //1.设置状态码为302
    resp.setStatus(302);
    //2.设置响应头location
    resp.setHeader("location", "/response/responseDemo2");
    
    //以上两步可以换成下面的语句
    resp.sendRedirect("/response/responseDemo2");
    
  • 重定向redirect和转发forward

    重定向redirect 转发forward
    地址栏发生变化:responseDemo1变成responseDemo2 转发地址栏路径不变
    重定向可以访问其他站点(服务器)的资源 转发只能访问当前服务器下的资源
    重定向是两次请求。
    先请求到Demo1,再链接到Demo2
    不能使用request对象来共享数据:
    转发是一次请求。可以使用request对象来共享数据
    //Demo1
    req.setAttribute("msg", "response");
    //Demo2
    Object msg = req.getAttribute("msg");
    System.out.println(msg);
    //重定向:不能共享数据--Demo2不能获取到msg的值
    //转发:共享数据
    
  • 路径写法:

    路径分类

    1. 相对路径:通过相对路径不可以确定唯一资源

如:./index.html
不以/开头,以.开头路径
规则:找到当前资源和目标资源之间的相对位置关系

  	./:当前目录;../:后退一级目录
  
  ```html
  <!DOCTYPE html>
  <html lang="en">
  <head>
      <meta charset="UTF-8">
      <title>Title</title>
  </head>
  <body>
      <h1>找到当前资源和目标资源之间的相对位置关系</h1>
      <p>
          当前资源:location.html
          http://localhost/response/location.html
      </p>
      <p>
          目标资源:
          http://localhost/response/responseDemo2
      </p>
      <a href="./responseDemo2">
          responseDemo2
      </a>
  </body>
  </html>
  ```


2. 绝对路径:通过绝对路径可以确定唯一资源

  如:http://localhost/response/responseDemo2		/response/responseDemo2

以/开头的路径
规则:判断定义的路径是给谁用的?判断请求将来从哪儿(客户端/服务器)发出

  1. 给客户端浏览器使用:需要加虚拟目录(项目的访问路径)
  
     建议虚拟目录动态获取:request.getContextPath()
     <a> , <form>  重定向...
  
  1. 给服务器使用:不需要加虚拟目录
     
     - 转发路径


案例二:服务器输出字符数据到浏览器

  • 步骤:
    1. 获取字符输出流
    2. 输出数据
  • 注意:

    乱码问题:

    PrintWriter pw = response.getWriter();获取的流的默认编码是ISO-8859-1

  1. 设置该流的默认编码:
    response.setCharacterEncoding(“utf-8”);
  2. 告诉浏览器响应体使用的编码:
    response.setHeader(“content-type”,“text/html;charset=utf-8”);
  3. //简单的形式,设置编码:
    response.setContentType(“text/html;charset=utf-8”);

案例三:服务器输出字节数据到浏览器

  • 步骤:
    1. 获取字节输出流
    2. 输出数据

补充案例:验证码

  1. 本质:图片
  2. 目的:防止恶意表单注册
package web.servlet;

@WebServlet("/checkCodeServlet")
public class CheckCodeServlet extends HttpServlet {
    @Override
    protected void doGet(HttpServletRequest req, HttpServletResponse resp) throws ServletException, IOException {
        this.doPost(req, resp);
    }

    @Override
    protected void doPost(HttpServletRequest req, HttpServletResponse resp) throws ServletException, IOException {
        int width = 100;
        int height = 50;

        //1.创建一个对象,在内存中图片(验证码对象),Ctrl+P查看方法参数
        BufferedImage image = new BufferedImage(width, height, BufferedImage.TYPE_INT_RGB);

        //2.美化图片
        //2.1 填充背景色
        Graphics g = image.getGraphics();//画笔对象
        g.setColor(Color.pink);//设置画笔颜色
        g.fillRect(0, 0, width, height);//填充颜色

        //2.2绘制边框
        g.setColor(Color.BLUE);
        g.drawRect(0, 0, width - 1, height - 1);//边框占一个像素,故-1

        String str = "ABCDEFGHIJKLMNOPQRSTUVWXYZabcdefghijklmnopqrstuvwxyz0123456789";
        //生成随机角标
        Random ran = new Random();

        //2.3写验证码
        for (int i = 1; i < 5; i++) {
            int index1 = ran.nextInt(str.length());
            char ch1 = str.charAt(index1);//随机字符,ch+""==转换成String
            g.drawString(ch1+"", width/5*i, height/2);
        }

        //2.4画干扰线
        g.setColor(Color.GREEN);//干扰线颜色
        for (int i = 1; i < 11; i++) {
            int x1 = ran.nextInt(width);
            int x2 = ran.nextInt(width);

            int y1 = ran.nextInt(height);
            int y2 = ran.nextInt(height);
            g.drawLine(x1, y1, x2, y2);

        }
        //3.将图片输出到页面显示
        ImageIO.write(image, "jpg", resp.getOutputStream());

    }
}

/*
        有了验证码后,更换时需要点击图片或者看不清换一张,这时候就需要绑定时间
        分析:点击超链接或者图片,需要换一张
            1.给超链接和图片绑定单击事件
            2.重新设置图片的src属性
         */

<!-- register.html -->
<!DOCTYPE html>
<html lang="en">
<head>
    <meta charset="UTF-8">
    <title>Title</title>
    <script>
        /*
        有了验证码后,更换时需要点击图片或者看不清换一张,这时候就需要绑定时间
        分析:点击超链接或者图片,需要换一张
            1.给超链接和图片绑定单击事件
            2.重新设置图片的src属性
         */
        window.onload=function () {
            //1.获取图片对象
            var img = document.getElementById("checkcode");
            //2.绑定单击事件
            img.onclick = function () {
                //加时间戳
                var date = new Date().getTime();
                img.src = "/response/checkCodeServlet?" + date;
            }
        }
    </script>
</head>
<body>
    <img id="checkcode" src="/response/checkCodeServlet"/>
    <a id="changecode" href="">看不清换一张</a>
</body>
</html>

三、ServletContext对象:

1-概念:

代表整个web应用,可以和**程序的容器(服务器)**来通信

2-获取:

  1. 通过request对象获取
    request.getServletContext();
  2. 通过HttpServlet获取
    this.getServletContext();
//1. 通过request对象获取
ServletContext context1 = req.getServletContext();
//2. 通过HttpServlet获取
ServletContext context2 = this.getServletContext();

System.out.println(context1);
System.out.println(context2);

System.out.println(context1 == context2);//true

3-功能:

1-获取MIME类型:

  • MIME类型:在互联网通信过程中定义的一种文件数据类型

  • 格式: 大类型/小类型 text/html image/jpeg

  • 获取:String getMimeType(String file)

    //2.通过HTTPServlet获取
    ServletContext context = this.getServletContext();
    //3.定义文件名称
    String filename = "a.jpg";
    //4.获取MIME类型
    String mimeType = context.getMimeType(filename);
    System.out.println(mimeType);
    

2-域对象:共享数据

  1. setAttribute(String name,Object value)
  2. getAttribute(String name)
  3. removeAttribute(String name)
  • ServletContext对象范围:所有用户所有请求的数据
//Demo3设置请求时间
//2.通过HTTPServlet获取
ServletContext context = this.getServletContext();
//设置数据
context.setAttribute("time","2020年2月9日12:09:48");
System.out.println("设置请求时间为:2020年2月9日12:09:48");

//Demo4通过另外一个客户端(浏览器)获取时间
//2.通过HTTPServlet获取
ServletContext context = this.getServletContext();
//获取数据
Object msg = context.getAttribute("time");
System.out.println(msg);

3-获取文件的真实(服务器)路径

  • 方法:String getRealPath(String path)

    // 通过HttpServlet获取
    ServletContext context = this.getServletContext();
    // 获取文件的服务器路径
    String b = context.getRealPath("/b.txt");//web目录下资源访问
    System.out.println(b);
    // File file = new File(realPath);
    
    String c = context.getRealPath("/WEB-INF/c.txt");//WEB-INF目录下的资源访问
    System.out.println(c);
    
    String a = context.getRealPath("/WEB-INF/classes/a.txt");//src目录下的资源访问
    System.out.println(a);
    

2020年2月9日12:31:08

案例:文件下载需求:

  1. 页面显示超链接
  2. 点击超链接后弹出下载提示框
  3. 完成图片文件下载

分析:

  1. 超链接指向的资源如果能够被浏览器解析,则在浏览器中展示,如果不能解析,则弹出下载提示框。不满足需求。
  2. 任何资源都必须弹出下载提示框。
  3. 使用响应头设置资源的打开方式:content-disposition:attachment;filename=xxx。

步骤:

  1. 定义页面,编辑超链接href属性,指向Servlet,传递资源名称filename

    <a href="/response/downloadServlet?filename=1.jpg">图片1</a>
    <a href="/response/downloadServlet?filename=1.jpg">视频1</a>
    
  2. 定义Servlet

  3. 获取文件名称

  4. 使用字节输入流加载文件进内存

  5. 指定response的响应头: content-disposition:attachment;filename=xxx

  6. 将数据写出到response输出流

//1.获取请求参数,文件名称
String filename = req.getParameter("filename");

//2.使用字节输入流加载文件进内存
//2.1找到文件服务器路径
ServletContext servletContext = this.getServletContext();
String realpath = servletContext.getRealPath("/img/" + filename);
//2.2用字节流关联
FileInputStream fis = new FileInputStream(realpath);

//3.设置response的响应头
//3.1设置响应头类型:content-type
String mimeType = servletContext.getMimeType(filename);//获取文件的mime类型
resp.setHeader("content-type", mimeType);

//3.2设置响应头打开方式:content-disposition
resp.setHeader("content-disposition", "attachment;filename=" + filename);

//        //解决中文文件名问题
//        //1.获取user-agent请求头
//        String agent = req.getHeader("user-agent");
//        //2.使用工具类方法编码文件名即可
//        filename = DownLoadUtils.getFileName(agent, filename);

//4.将输入流的数据写出到输出流中
ServletOutputStream sos = resp.getOutputStream();

byte[] buff=new byte[1024*8];
int len = 0;
while ((len=fis.read(buff))!=-1){
    sos.write(buff, 0, len);
}
fis.close();

问题:

  • 中文文件问题

    工具类Utils不需要自己写!!!

    • 解决思路:
      1. 获取客户端使用的浏览器版本信息
      2. 根据不同的版本信息,设置filename的编码方式不同
发布了92 篇原创文章 · 获赞 23 · 访问量 6万+

猜你喜欢

转载自blog.csdn.net/Pit3369/article/details/104248384