只需这一篇文章!彻底帮你学会Response响应

489 阅读7分钟

前言

功能:设置响应消息
	1.设置响应行
		1.格式:HTTP/1.1 200 ok
		2.设置状态码:setStatus(int sc)
	2.设置响应头:setHeader(String name,String value)

	3.设置响应体:
		使用步骤:
			1.获取输出流
				字符输出流:PrintWriter getWriter()
				字节输出流:ServletOutputStream getOutputStream()
			2.使用输出流,将数据输出到客户端浏览器中

案例步骤:

1.完成重定向
	重定向:资源跳转的方式
	代码实现:
        //1.设置状态码为302
        response.setStatus(302);
        //2.设置响应头location
        response.setHeader("location","/day15/responseDemo2");

        //简单的重定向方法
        response.sendRedirect("/day15/responseDemo2");

    重定向的特点:redirect
    	1.地址栏发生变化
    	2.重定向可以访问其他站点(服务器)的资源
    	3.重定向是两次请求,不能使用request对象来共享数据
    转发的特点:forward
    	1.转发地址栏路径不变
    	2.转发只能访问当前服务器下的资源
    	3.转发是一次请求,可以使用request对象共享数据
    路径写法:
    	1.路径分类
    		1.相对路径:通过相对路径不可以确定唯一资源
    			如:./index.html
    			不以/开头,以.开头的路径

				规则:确定访问的当前资源和目标资源之间的相对位置关系
					./:表示当前目录
					../:后退一级目录
    		2.绝对路径:通过绝对路径可以确定唯一资源
    			如:http://localhost/day15/responseDemo2
    			以/开头的路径

				规则:判断定义的路径是给谁用的?判断请求将来从哪发出
					给客户端浏览器使用:需要加虚拟目录(项目的访问路径)
						介意虚拟目录动态获取:request.getContextPath()
					给服务器使用:不需要加虚拟目录
						转发路径

2.服务器输出字符数据到浏览器
	步骤:
		1.获取字符输出流
		2.输出数据
	注意:
		乱码问题:
			1.PrintWriter pw = response.getWriter();获取的流的默认编码是ISO-88859-1
			2.设置该的默认编码
			3.告诉浏览器响应体使用的编码

		//简单的形式,设置编码,是在获取流之前设置
		response.setContentType("text/html;Charset=utf-8");

3.服务器输出字节数据到浏览器	
	步骤:
  		1.获取字节输出流
		2.输出数据

4.验证码
	1.本质:图片
	2.目的:放置恶意表单注册

案例1:重定向代码实现

重定向图解:红线表示的就是重定向

代码实现

package cn.itcast.web.servlet;

import javax.servlet.ServletException;
import javax.servlet.annotation.WebServlet;
import javax.servlet.http.HttpServlet;
import javax.servlet.http.HttpServletRequest;
import javax.servlet.http.HttpServletResponse;
import java.io.IOException;

/**
 * 完成重定向
 */
@WebServlet("/responseDemo1")
public class ResponseDemo1 extends HttpServlet {
    protected void doPost(HttpServletRequest request, HttpServletResponse response) throws ServletException, IOException {
        System.out.println("demo1........");

        /*//访问/responseDemo1资源,会自动跳转到/responseDemo2资源
        //1.设置状态码为302
        response.setStatus(302);
        //2.设置响应头location
        response.setHeader("location","/day15/responseDemo2");*/

        //简单的重定向方法
        response.sendRedirect("/day15/responseDemo2");
    }

    protected void doGet(HttpServletRequest request, HttpServletResponse response) throws ServletException, IOException {
        this.doPost(request,response);
    }
}

=========================================================================================
package cn.itcast.web.servlet;

import javax.servlet.ServletException;
import javax.servlet.annotation.WebServlet;
import javax.servlet.http.HttpServlet;
import javax.servlet.http.HttpServletRequest;
import javax.servlet.http.HttpServletResponse;
import java.io.IOException;

@WebServlet("/responseDemo2")
public class ResponseDemo2 extends HttpServlet {
    protected void doPost(HttpServletRequest request, HttpServletResponse response) throws ServletException, IOException {
        System.out.println("demo2......");
    }

    protected void doGet(HttpServletRequest request, HttpServletResponse response) throws ServletException, IOException {
        this.doPost(request,response);
    }
}

案例2:输出字符数据

不墨迹,直接看代码:

package cn.itcast.web.servlet;

import javax.servlet.ServletException;
import javax.servlet.annotation.WebServlet;
import javax.servlet.http.HttpServlet;
import javax.servlet.http.HttpServletRequest;
import javax.servlet.http.HttpServletResponse;
import java.io.IOException;
import java.io.PrintWriter;

@WebServlet("/responseDemo4")
public class ResponseDemo4 extends HttpServlet {
    protected void doPost(HttpServletRequest request, HttpServletResponse response) throws ServletException, IOException {
        //1.获取字符输出流
        PrintWriter pw = response.getWriter();
        //2.输出数据
        pw.write("<h1>hello response</h1>");
    }

    protected void doGet(HttpServletRequest request, HttpServletResponse response) throws ServletException, IOException {
        this.doPost(request,response);
    }
}

debug方式启动后,直接在页面上显示:

很明显,输出字符数据真的非常的简单。

接下来讲一个比较麻烦的事:中文乱码的问题 因为我们输出的数据在所难免会遇到中文的数据。代码如下:

//2.输出数据
pw.write("你好,response");

这样就会出现乱码情况:

在这里插入图片描述

那么接下来我们分析一下乱码的原因:无外乎就是编码和解码的码表不一致。

如上图所示: 浏览器打开默认的字符集跟当前操作系统的语言和环境有关系,我们现在用的windows操作系统默认的字符集编码表是GBK(gb2312) 服务器用的不是GBK,否则不会乱码。

字符输出流的编码是ISO的,该编码被GBK解码一定会乱码。 找到原因之后,解决这个乱码就会变得很简单:我们可以设置流的编码是GBK就可以了。

package cn.itcast.web.servlet;

import javax.servlet.ServletException;
import javax.servlet.annotation.WebServlet;
import javax.servlet.http.HttpServlet;
import javax.servlet.http.HttpServletRequest;
import javax.servlet.http.HttpServletResponse;
import java.io.IOException;
import java.io.PrintWriter;

@WebServlet("/responseDemo4")
public class ResponseDemo4 extends HttpServlet {
    protected void doPost(HttpServletRequest request, HttpServletResponse response) throws ServletException, IOException {
        //获取流对象之前去设置流的默认编码:ISO-8859-1 设置为:GBK
        response.setCharacterEncoding("GBK");
        //1.获取字符输出流
        PrintWriter pw = response.getWriter();
        //2.输出数据
        pw.write("你好,response");
    }

    protected void doGet(HttpServletRequest request, HttpServletResponse response) throws ServletException, IOException {
        this.doPost(request,response);
    }
}

设置完流的默认编码后刷新一下服务器就发现,可以识别中文了:

但是仅仅是一种巧合,为什么? 因为我们明确的知道客户端用的是GBK才可以这样设置。将来有的浏览器就是utf-8,就意味着设置GBK不好使。 我们还需要告诉浏览器,服务器发送的消息体数据的编码,建议浏览器使用该编码解码。如何完成代码如下:

package cn.itcast.web.servlet;

import javax.servlet.ServletException;
import javax.servlet.annotation.WebServlet;
import javax.servlet.http.HttpServlet;
import javax.servlet.http.HttpServletRequest;
import javax.servlet.http.HttpServletResponse;
import java.io.IOException;
import java.io.PrintWriter;
import java.nio.charset.Charset;

@WebServlet("/responseDemo4")
public class ResponseDemo4 extends HttpServlet {
    protected void doPost(HttpServletRequest request, HttpServletResponse response) throws ServletException, IOException {
        //获取流对象之前去设置流的默认编码:ISO-8859-1 设置为:GBK
        //response.setCharacterEncoding("utf-8");
        //告诉浏览器,服务器发送的消息体数据的编码。建议浏览器使用该编码解码
        response.setHeader("content-type","text/html;Charset=utf-8");
        //1.获取字符输出流
        PrintWriter pw = response.getWriter();
        //2.输出数据
        pw.write("你好,response");
    }

    protected void doGet(HttpServletRequest request, HttpServletResponse response) throws ServletException, IOException {
        this.doPost(request,response);
    }
}

但是这段代码写起来还是挺麻烦的,因为头信息是固定的,也就是说每一次都要写这行代码。所以response对象给我们提供的一个简单的型式来设置编码:

package cn.itcast.web.servlet;

import javax.servlet.ServletException;
import javax.servlet.annotation.WebServlet;
import javax.servlet.http.HttpServlet;
import javax.servlet.http.HttpServletRequest;
import javax.servlet.http.HttpServletResponse;
import java.io.IOException;
import java.io.PrintWriter;
import java.nio.charset.Charset;

@WebServlet("/responseDemo4")
public class ResponseDemo4 extends HttpServlet {
    protected void doPost(HttpServletRequest request, HttpServletResponse response) throws ServletException, IOException {
        //获取流对象之前去设置流的默认编码:ISO-8859-1 设置为:GBK
        response.setCharacterEncoding("utf-8");
        //告诉浏览器,服务器发送的消息体数据的编码。建议浏览器使用该编码解码
        //response.setHeader("content-type","text/html;Charset=utf-8");
        //简单的形式,设置编码
        response.setContentType("text/html;Charset=utf-8");
        //1.获取字符输出流
        PrintWriter pw = response.getWriter();
        //2.输出数据
        pw.write("你好啊啊啊啊,response");
    }

    protected void doGet(HttpServletRequest request, HttpServletResponse response) throws ServletException, IOException {
        this.doPost(request,response);
    }
}

以后我们要记住,在获取字符输出流并且写中文数据之前,就要把这行代码写到最前面:

response.setContentType("text/html;Charset=utf-8");

案例3:输出字节数据

接下来说,输出字节数据,代码如下:

package cn.itcast.web.servlet;

import javax.servlet.ServletException;
import javax.servlet.ServletOutputStream;
import javax.servlet.annotation.WebServlet;
import javax.servlet.http.HttpServlet;
import javax.servlet.http.HttpServletRequest;
import javax.servlet.http.HttpServletResponse;
import java.io.IOException;

@WebServlet("/responseDemo5")
public class ResponseDemo5 extends HttpServlet {
    protected void doPost(HttpServletRequest request, HttpServletResponse response) throws ServletException, IOException {
        //1.获取字节输出流
        ServletOutputStream sos = response.getOutputStream();
        //2.输出数据
        sos.write("hello".getBytes());
    }

    protected void doGet(HttpServletRequest request, HttpServletResponse response) throws ServletException, IOException {
        this.doPost(request,response);
    }
}

hello就被访问到了:

当我们想输出中文,并且设置的是utf-8就会乱码:

//2.输出数据
sos.write("你好".getBytes("utf-8"));

解决方式:

response.setHeader("content-type","text/html;Charset=utf-8");

案例4:验证码

创建验证码步骤非常简单,我们直接代码演示:

package cn.itcast.web.servlet;

import javax.imageio.ImageIO;
import javax.servlet.ServletException;
import javax.servlet.ServletOutputStream;
import javax.servlet.annotation.WebServlet;
import javax.servlet.http.HttpServlet;
import javax.servlet.http.HttpServletRequest;
import javax.servlet.http.HttpServletResponse;
import java.awt.*;
import java.awt.image.BufferedImage;
import java.io.IOException;
import java.nio.Buffer;
import java.security.DigestOutputStream;
import java.util.Random;

@WebServlet("/checkCodeServlet")
public class CheckCodeServlet extends HttpServlet {
    protected void doPost(HttpServletRequest request, HttpServletResponse response) throws ServletException, IOException {
        int width = 100;
        int height = 50;
        // 1.创建一个对象,在内存中画图(验证码图片对象)
        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);

        String str = "ABCDEFGHIJKLMNOPQRSTUVWXYZabcdefghijklmnopqrstuvwxyz0123456789";

        // 生成随机角标
        Random ran = new Random();

        for (int i = 1; i <= 4; i++) {
            int index = ran.nextInt(str.length());
            // 获取字符
            char ch = str.charAt(index); // 随机字符
            // 2.3写验证码
            g.drawString(ch + "" ,width / 5 * i,height / 2);
        }

        // 2.4画干扰线
        g.setColor(Color.green);

        // 随机生成坐标点
        for (int i = 0; i < 10; 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",response.getOutputStream());
    }

    protected void doGet(HttpServletRequest request, HttpServletResponse response) throws ServletException, IOException {
        this.doPost(request,response);
    }
}

代码效果如下,重新加载会呈现不同的验证码:

学习完了使用代码的方式来实现验证码,接下来,说一下真正将来在程序开发过程中如何使用验证码,当然不是在页面上显示一个验证码,是配合注册表单一起来显示验证码。(只能是一张图片)

接下来说一下验证码切换效果: 我们写一个HTML页面,代码如下:

<!DOCTYPE html>
<html lang="en">
<head>
    <meta charset="UTF-8">
    <title>Title</title>
    <script>
        /*
            分析:
                点击超链接或者图片,需要换一张
                1.给超链接和图片绑定单击事件
                2.重新设置图片的src属性值
         */
     window.onload = function () {
         // 1.获取图片对象
         let img = document.getElementById("checkCode");
         // 2.绑定单击事件
         img.onclick = function () {
             // 加时间戳
             var date = new Date().getTime();
            img.src = "/day15/checkCodeServlet?" + date;
         }
     }

    </script>
</head>
<body>
    <img id="checkCode" src="/day15/checkCodeServlet" alt="">
    <a id="change" href="">看不清换一张?</a>
</body>
</html>

最后

感谢你看到这里,文章有什么不足还请指正,觉得文章对你有帮助的话记得给我点个赞,每天都会分享java相关技术文章或行业资讯,欢迎大家关注和转发文章!