Request和Response的概述,能够完成SqlSessionFactory工具类的抽取(中)

简介: Request和Response的概述,能够完成SqlSessionFactory工具类的抽取

3,Response对象


前面讲解完Request对象,接下来我们回到刚开始的那张图:


1e501d07f99d47e09b976e3733a79954.png


Request:使用request对象来获取请求数据

Response:使用response对象来设置响应数据

Reponse的继承体系和Request的继承体系也非常相似:


0a5d9e8ddb424eddb73be4d1889ab98d.png


介绍完Response的相关体系结构后,接下来对于Response我们需要学习如下内容:


Response设置响应数据的功能介绍

Response完成重定向

Response响应字符数据

Response响应字节数据

3.1 Response设置响应数据功能介绍

HTTP响应数据总共分为三部分内容,分别是响应行、响应头、响应体,对于这三部分内容的数据,respone对象都提供了哪些方法来进行设置?


响应行


333beb53ab264c3293d7403a13f49570.png


对于响应头,比较常用的就是设置响应状态码:


voidsetStatus(intsc);


响应头


4889b04b04194620be940ead1a390bad.png


设置响应头键值对:


voidsetHeader(Stringname,Stringvalue);


响应体


70140b75786f4bd9b6a8521cbdbd3ee7.png


对于响应体,是通过字符、字节输出流的方式往浏览器写,


获取字符输出流:


PrintWritergetWriter();


获取字节输出流


ServletOutputStreamgetOutputStream();


介绍完这些方法后,后面我们会通过案例把这些方法都用一用,首先先来完成下重定向的功能开发。


3.2 Respones请求重定向

Response重定向(redirect):一种资源跳转方式。


505e71cb3e214ea7bed80019d6d64882.png


(1)浏览器发送请求给服务器,服务器中对应的资源A接收到请求


(2)资源A现在无法处理该请求,就会给浏览器响应一个302的状态码+location的一个访问资源B的路径


(3)浏览器接收到响应状态码为302就会重新发送请求到location对应的访问地址去访问资源B


(4)资源B接收到请求后进行处理并最终给浏览器响应结果,这整个过程就叫重定向


重定向的实现方式:

resp.setStatus(302);
resp.setHeader("location","资源B的访问路径");


具体如何来使用,我们先来看下需求:


7fd3cc9222fe45eabf28dcbe80f1fe69.png


针对上述需求,具体的实现步骤为:


1.创建一个ResponseDemo1类,接收/resp1的请求,在doGet方法中打印resp1....


2.创建一个ResponseDemo2类,接收/resp2的请求,在doGet方法中打印resp2....


3.在ResponseDemo1的方法中使用


response.setStatus(302);


response.setHeader(“Location”,“/request-demo/resp2”) 来给前端响应结果数据


4.启动测试


(1)创建ResponseDemo1类


@WebServlet("/resp1")
publicclassResponseDemo1extendsHttpServlet {
@OverrideprotectedvoiddoGet(HttpServletRequestrequest, HttpServletResponseresponse) throwsServletException, IOException {
System.out.println("resp1....");
    }
@OverrideprotectedvoiddoPost(HttpServletRequestrequest, HttpServletResponseresponse) throwsServletException, IOException {
this.doGet(request, response);
    }
}


(2)创建ResponseDemo2类


@WebServlet("/resp2")
publicclassResponseDemo2extendsHttpServlet {
@OverrideprotectedvoiddoGet(HttpServletRequestrequest, HttpServletResponseresponse) throwsServletException, IOException {
System.out.println("resp2....");
    }
@OverrideprotectedvoiddoPost(HttpServletRequestrequest, HttpServletResponseresponse) throwsServletException, IOException {
this.doGet(request, response);
    }
}


(3)在ResponseDemo1的doGet方法中给前端响应数据


@WebServlet("/resp1")
publicclassResponseDemo1extendsHttpServlet {
@OverrideprotectedvoiddoGet(HttpServletRequestrequest, HttpServletResponseresponse) throwsServletException, IOException {
System.out.println("resp1....");
//重定向//1.设置响应状态码 302response.setStatus(302);
//2. 设置响应头 Locationresponse.setHeader("Location","/request-demo/resp2");
    }
@OverrideprotectedvoiddoPost(HttpServletRequestrequest, HttpServletResponseresponse) throwsServletException, IOException {
this.doGet(request, response);
    }
}



(4)启动测试


访问http://localhost:8080/request-demo/resp1,就可以在控制台看到如下内容:


789944eecf9b4569ae0d27df79eec4fd.png


说明/resp1和/resp2都被访问到了。到这重定向就已经完成了。


虽然功能已经实现,但是从设置重定向的两行代码来看,会发现除了重定向的地址不一样,其他的内容都是一模一样,所以request对象给我们提供了简化的编写方式为:


resposne.sendRedirect("/request-demo/resp2")


所以第3步中的代码就可以简化为:


@WebServlet("/resp1")
publicclassResponseDemo1extendsHttpServlet {
@OverrideprotectedvoiddoGet(HttpServletRequestrequest, HttpServletResponseresponse) throwsServletException, IOException {
System.out.println("resp1....");
//重定向resposne.sendRedirect("/request-demo/resp2")    }
@OverrideprotectedvoiddoPost(HttpServletRequestrequest, HttpServletResponseresponse) throwsServletException, IOException {
this.doGet(request, response);
    }
}


重定向的特点

浏览器地址栏路径发送变化


当进行重定向访问的时候,由于是由浏览器发送的两次请求,所以地址会发生变化


可以重定向到任何位置的资源(服务内容、外部均可)


因为第一次响应结果中包含了浏览器下次要跳转的路径,所以这个路径是可以任意位置资源。


两次请求,不能在多个资源使用request共享数据


因为浏览器发送了两次请求,是两个不同的request对象,就无法通过request对象进行共享数据


介绍完请求重定向和请求转发以后,接下来需要把这两个放在一块对比下:


以后到底用哪个,还是需要根据具体的业务来决定。


3.3 路径问题

问题1:转发的时候路径上没有加/request-demo而重定向加了,那么到底什么时候需要加,什么时候不需要加呢?

其实判断的依据很简单,只需要记住下面的规则即可:


浏览器使用:需要加虚拟目录(项目访问路径)

服务端使用:不需要加虚拟目录

对于转发来说,因为是在服务端进行的,所以不需要加虚拟目录


对于重定向来说,路径最终是由浏览器来发送请求,就需要添加虚拟目录。


掌握了这个规则,接下来就通过一些练习来强化下知识的学习:


req.getRequestDispatcher(“路径”)

resp.sendRedirect(“路径”)

答案:


1.超链接,从浏览器发送,需要加

2.表单,从浏览器发送,需要加

3.转发,是从服务器内部跳转,不需要加

4.重定向,是由浏览器进行跳转,需要加。

1

2

3

4

问题2:在重定向的代码中,/request-demo是固定编码的,如果后期通过Tomcat插件配置了项目的访问路径,那么所有需要重定向的地方都需要重新修改,该如何优化?

答案也比较简单,我们可以在代码中动态去获取项目访问的虚拟目录,具体如何获取,我们可以借助前面咱们所学习的request对象中的getContextPath()方法,修改后的代码如下:


@WebServlet("/resp1")
publicclassResponseDemo1extendsHttpServlet {
@OverrideprotectedvoiddoGet(HttpServletRequestrequest, HttpServletResponseresponse) throwsServletException, IOException {
System.out.println("resp1....");
//简化方式完成重定向//动态获取虚拟目录StringcontextPath=request.getContextPath();
response.sendRedirect(contextPath+"/resp2");
    }
@OverrideprotectedvoiddoPost(HttpServletRequestrequest, HttpServletResponseresponse) throwsServletException, IOException {
this.doGet(request, response);
    }
}



重新启动访问测试,功能依然能够实现,此时就可以动态获取项目访问的虚拟路径,从而降低代码的耦合度。


3.4 Response响应字符数据

要想将字符数据写回到浏览器,我们需要两个步骤:


通过Response对象获取字符输出流: PrintWriter writer = resp.getWriter();


通过字符输出流写数据: writer.write(“aaa”);


接下来,我们实现通过些案例把响应字符数据给实际应用下:


返回一个简单的字符串aaa

/*** 响应字符数据:设置字符数据的响应体*/@WebServlet("/resp3")
publicclassResponseDemo3extendsHttpServlet {
@OverrideprotectedvoiddoGet(HttpServletRequestrequest, HttpServletResponseresponse) throwsServletException, IOException {
response.setContentType("text/html;charset=utf-8");
//1. 获取字符输出流PrintWriterwriter=response.getWriter();
writer.write("aaa");
    }
@OverrideprotectedvoiddoPost(HttpServletRequestrequest, HttpServletResponseresponse) throwsServletException, IOException {
this.doGet(request, response);
    }
}


返回一串html字符串,并且能被浏览器解析

PrintWriterwriter=response.getWriter();
//content-type,告诉浏览器返回的数据类型是HTML类型数据,这样浏览器才会解析HTML标签response.setHeader("content-type","text/html");
writer.write("<h1>aaa</h1>");


0c6877a373814f12bc0e107f6789d6c4.png


==注意:==一次请求响应结束后,response对象就会被销毁掉,所以不要手动关闭流。


返回一个中文的字符串你好,需要注意设置响应数据的编码为utf-8

//设置响应的数据格式及数据的编码

response.setContentType("text/html;charset=utf-8");
writer.write("你好");


b424901a7b1a4e1493a24c06edea2f91.png


3.3 Response响应字节数据

要想将字节数据写回到浏览器,我们需要两个步骤:


通过Response对象获取字节输出流:ServletOutputStream outputStream = resp.getOutputStream();


通过字节输出流写数据: outputStream.write(字节数据);


接下来,我们实现通过些案例把响应字符数据给实际应用下:


返回一个图片文件到浏览器

/*** 响应字节数据:设置字节数据的响应体*/@WebServlet("/resp4")
publicclassResponseDemo4extendsHttpServlet {
@OverrideprotectedvoiddoGet(HttpServletRequestrequest, HttpServletResponseresponse) throwsServletException, IOException {
//1. 读取文件FileInputStreamfis=newFileInputStream("d://a.jpg");
//2. 获取response字节输出流ServletOutputStreamos=response.getOutputStream();
//3. 完成流的copybyte[] buff=newbyte[1024];
intlen=0;
while ((len=fis.read(buff))!=-1){
os.write(buff,0,len);
        }
fis.close();
    }
@OverrideprotectedvoiddoPost(HttpServletRequestrequest, HttpServletResponseresponse) throwsServletException, IOException {
this.doGet(request, response);
    }
}



上述代码中,对于流的copy的代码还是比较复杂的,所以我们可以使用别人提供好的方法来简化代码的开发,具体的步骤是:


(1)pom.xml添加依赖


<dependency><groupId>commons-io</groupId><artifactId>commons-io</artifactId><version>2.6</version></dependency>


(2)调用工具类方法


//fis:输入流//os:输出流IOUtils.copy(fis,os);


优化后的代码:


/*** 响应字节数据:设置字节数据的响应体*/@WebServlet("/resp4")
publicclassResponseDemo4extendsHttpServlet {
@OverrideprotectedvoiddoGet(HttpServletRequestrequest, HttpServletResponseresponse) throwsServletException, IOException {
//1. 读取文件FileInputStreamfis=newFileInputStream("d://a.jpg");
//2. 获取response字节输出流ServletOutputStreamos=response.getOutputStream();
//3. 完成流的copyIOUtils.copy(fis,os);
fis.close();
    }
@OverrideprotectedvoiddoPost(HttpServletRequestrequest, HttpServletResponseresponse) throwsServletException, IOException {
this.doGet(request, response);
    }
}
相关文章
|
存储 算法 搜索推荐
【潜意识Java】期末考试可能考的高质量大题及答案解析
Java 期末考试大题整理:设计一个学生信息管理系统,涵盖面向对象编程、集合类、文件操作、异常处理和多线程等知识点。系统功能包括添加、查询、删除、显示所有学生信息、按成绩排序及文件存储。通过本题,考生可以巩固 Java 基础知识并掌握综合应用技能。代码解析详细,适合复习备考。
256 4
|
12月前
|
消息中间件 人工智能 Kafka
【云故事探索】NO.14:乐言科技——云原生加速电商行业赋能,云消息队列助力降本 37%
上海乐言科技股份有限公司专注于AI技术,提供电商、金融等领域的整体解决方案。其核心产品“乐语助人”智能客服机器人日均服务超千万人次,助力六万余家电商客户数智化转型。为解决自建消息队列痛点,乐言科技采用阿里云消息队列RocketMQ版Serverless系列,实现业务稳定、开发成本降低、运维效率提升及资源弹性降本37%。通过云原生架构,乐言科技推动AI与电商深度融合,助力行业创新突破。
|
人工智能 分布式计算 DataWorks
大数据& AI 产品月刊【2025年1、2月】
大数据& AI 产品技术月刊【2025年1、2月】,涵盖双月技术速递、产品和功能发布、市场和客户应用实践等内容,帮助您快速了解阿里云大数据& AI 方面最新动态。
|
敏捷开发 数据可视化 搜索推荐
项目管理看板:项目进度的清晰导航
项目管理看板是一种可视化的任务管理工具,起源于日本丰田公司的精益生产方法。它通过分阶段展示任务状态,帮助团队实时跟踪进展,提高协作效率。看板广泛应用于软件开发、营销、产品开发和客户服务等领域,核心功能包括可视化任务管理、实时跟踪、提高协作、标识阻塞问题和数据分析。未来,看板将更加智能化和集成化,支持更多自定义功能。
|
存储 Java
抽象类与抽象方法
抽象类和抽象方法是面向对象编程中的重要概念。抽象类不能被实例化,主要用于定义子类的通用接口或行为。抽象方法是在抽象类中声明但未实现的方法,要求子类必须实现这些方法。这种方式有助于构建灵活且可扩展的代码结构。
432 2
|
Unix Linux Shell
linux入门!
本文档介绍了Linux系统入门的基础知识,包括操作系统概述、CentOS系统的安装与远程连接、文件操作、目录结构、用户和用户组管理、权限管理、Shell基础、输入输出、压缩打包、文件传输、软件安装、文件查找、进程管理、定时任务和服务管理等内容。重点讲解了常见的命令和操作技巧,帮助初学者快速掌握Linux系统的基本使用方法。
849 3
|
存储 监控 安全
SaaS业务架构:业务能力分析
【9月更文挑战第20天】在数字化时代,软件即服务(SaaS)模式逐渐成为企业软件解决方案的首选。SaaS 业务架构设计对于提供高效、可靠的服务至关重要。其核心业务能力包括:用户管理(注册登录、角色权限)、数据管理(存储备份、安全共享)、业务流程管理(设计定制、工作流自动化)、应用集成(第三方应用、移动应用)及客户服务(支持培训、反馈改进)。通过优化这些能力,可为企业提供更高效、可靠的 SaaS 服务。
382 11
|
运维 负载均衡 Devops
DevOps实践:使用Ansible进行自动化部署
【9月更文挑战第19天】在软件开发的快节奏世界中,DevOps已成为提高效率和促进协作的关键。本文将通过一个实际案例,展示如何使用Ansible简化自动化部署过程,旨在帮助读者理解DevOps的核心价值并掌握Ansible的基本使用。从概念到实践,我们将一起探索如何通过编写简单的Playbook来自动化服务器配置和应用部署,最终实现快速、可靠的软件发布。
|
机器学习/深度学习 自然语言处理 运维
探索深度学习在图像识别中的最新进展
探索深度学习在图像识别中的最新进展
|
存储 人工智能 安全
Gartner预测2025年IT支出将大幅增加
Gartner预测2025年IT支出将大幅增加

热门文章

最新文章