Servlet详解
一.Servlet生命周期
- 初始化
- 提供服务
- 销毁
1.测试生命周期
package com.demo.servlet;import javax.servlet.*;
import java.io.IOException;public class LifeServlet implements Servlet {@Overridepublic void init(ServletConfig servletConfig) throws ServletException {System.out.println("Servlet init");}@Overridepublic void service(ServletRequest servletRequest, ServletResponse servletResponse)throws ServletException, IOException {System.out.println("Servlet service");}@Overridepublic void destroy() {System.out.println("Servlet destroy");}@Overridepublic ServletConfig getServletConfig() {return null;}@Overridepublic String getServletInfo() {return null;}
}
2.初始化:init()
(1)默认情况
- 默认情况下,第一次被请求的时候初始化,初始化在整个生命中只有一次
(2)设置初始化时间
-
某种原因,导致第一次请求时初始化很慢,这时就可以修改Servlet的初始化时间
-
使用其值为大于等于0的整数时,该Servlet就会在服务器启动时初始化,初始化在整个生命中只有一次
3.提供服务:service()
- 生命周期中会提供n次
4.销毁:destroy()
- 服务器关闭时自动销毁,生命周期中只有一次
二.Servlet的映射路径
1.web.xml配置
(1)标签
- 的子标签,用于给Servlet配置访问路径
(2)注意
- 路径最前面加上/,代表根目录
- 路径中可以出现文件夹的,如/user/test/urlServlet
(3)示例
- http://localhost:8888/day02_war_exploded/user/test/lifeServlet
- http://localhost:8888/day02_war_exploded/user/life.html
2.模块前端文件的访问路径
- 访问文件时,只需要关注自己创建的目录,自动生成的不管
- 访问Servlet时,照着url-pattern设置的值来找
3.Servlet3.0注解配置
(1)使用前提
- Servlet3.0和JDK1.5及以后的版本才能使用
(2)使用
(3)注意
-
同一个Servlet,要么使用web.xml文件配置,要么使用注解配置,但是不能使用两种方式同时配置
-
注解中的参数有name、serviceName;若是只写name参数,可以省略key值,只写value;除此之外要写key-value的形式
三.Servlet请求对象
1.HttpServletRequest接口
- 请求对象
- 父接口:ServletRequest
- 包含请求行,请求头和请求体
2.请求行
(1)包含
- 请求方式
- 请求的资源
- 协议及其版本
(2)获取信息
@Overrideprotected void doGet(HttpServletRequest req, HttpServletResponse resp)throws ServletException, IOException {System.out.println(req.getMethod());System.out.println(req.getRequestURL());System.out.println(req.getRequestURI());System.out.println(req.getServerName());System.out.println(req.getServerPort());}
3.请求头
(1)获取信息
protected void doGet(HttpServletRequest req, HttpServletResponse resp)throws ServletException, IOException {String value1 = req.getHeader("Host");System.out.println(value1);Enumeration<String> names = req.getHeaderNames();while (names.hasMoreElements()) {String name = names.nextElement();String value = req.getHeader(name);System.out.println(name+":"+value);}}
4.请求体
- 就是请求的数据,也就是Servlet中获取的参数
(1)Servlet中获取请求的数据的方法
- request.getParameter(“name属性的值”);返回值类型为String
- request.getParameterValues(“name属性的值”);返回值类型为String[]
(2)示例
@Overrideprotected void doPost(HttpServletRequest req, HttpServletResponse resp)throws ServletException, IOException {String username = req.getParameter("username");String password = req.getParameter("password");String gender = req.getParameter("gender");String hobbies[] = req.getParameterValues("hobbies");System.out.println(username+","+password+","+gender+","+ Arrays.toString(hobbies));}
5.Servlet获取参数乱码问题
(1)问题
- 中文乱码,Servlet的默认编码集是ISO-8859-1(纯英文)
- 页面输入中文进行提交,会乱码
(2)解决方案
四.文件上传
- 就是在指定的位置复制出一个跟源文件一模一样的文件
1.页面编写
- 表单的method属性的值必须是pos
- 表单上加上enctype,值为multipart/form-data
2.Servlet
- 在Servlet上加上@MultipartConfig
3.Post方法编写
- 不同浏览器上传文件的路径不一样
- 绝对路径(不带盘符)
- 文件名+文件类型后缀
@Overrideprotected void doPost(HttpServletRequest req, HttpServletResponse resp)throws ServletException, IOException {//设置请求体编码集req.setCharacterEncoding("utf-8");String username = req.getParameter("username");String password = req.getParameter("password");String gender = req.getParameter("gender");String hobbies[] = req.getParameterValues("hobbies");System.out.println(username+","+password+","+gender+","+ Arrays.toString(hobbies));//文件相关处理//获取用户提交的文件对象Part part = req.getPart("photo");//指定上传路径String path = "d://upload";//判断该路径是否存在File file = new File(path);if (!file.exists()) {//不存在则创建file.mkdirs();}//获取上传的文件的名字String fileName = part.getSubmittedFileName();//在指定位置创建指定文件part.write(path+"/"+fileName);}
五.Servlet响应对象
1.HttpServletResponse接口
- 服务器响应对象
- 父接口:ServletResponse
- 包含响应行,响应头和响应体。
2.响应行
- 包含协议及其版本,状态码
(1)状态码
状态码 | 说明 |
---|---|
1** | 请求已到达服务器,但未完成,需要再次请求 |
2** | 请求已到达服务器,并且已完成 |
3** | 服务器内部资源重定向 |
4** | 资源找不到 |
5** | 服务器内部异常,代码异常 |
200 | 成功 |
404 | 找不到,原因:路径有问题 |
500 | 代码有问题,90%以上都是空指针 |
(2)设置状态码
@Overrideprotected void doGet(HttpServletRequest req, HttpServletResponse resp)throws ServletException, IOException {resp.setStatus(404);}
3.响应头
响应头key | value |
---|---|
Location | 这个头通常配合302状态码使用,它用于告诉浏览器你去找谁。 |
Server | 告诉浏览器,服务器的类型 |
Content-Encoding | 服务器通过这个头,告诉浏览器,回送的数据采用的压缩格式。 |
Content-Length | 0 |
Content-Language | zh-cn |
Content-Type | 这个头用于告诉浏览器,回送数据的类型 |
Last-Modified | 这个头用于告诉浏览器,数据的最后修改时间 |
Transfer-Encoding | 用于通知浏览器,数据是以分块形式回送的 |
Cache-Control | no-cache 通知浏览器不要缓存 |
Refresh | 1;url=http://www.it315.org 隔多少秒以后,让当前页面去访问哪个地址(例如网页登陆成功后,跳回原来的界面,就是用的这个头) |
Content-Disposition | attachment;filename=aaa.zip 和下载相关,通知浏览器以附件的形式下载服务器发送过去的数据 |
Set-Cookie | SS=Q0=5Lb_nQ; path=/search 和cookie相关的头 |
ETag | W/“83794-1208174400000” 和cookie相关的头 |
Expires | -1 通知浏览器是否缓存当前资源:如果这个头的值是一个以毫秒为单位值,则通知浏览器缓存资源到指定的时间点;如果值是0或-1,则通知浏览器禁止缓存 |
Pragma | no-cache 通知浏览器是否缓存资源 |
Connection | close/Keep-Alive 是否继续保持连接 |
Date | Tue, 11 Jul 2000 18:23:51 GMT 当前响应的时间 |
(1)Location的使用
protected void doGet(HttpServletRequest req, HttpServletResponse resp)throws ServletException, IOException {resp.setStatus(302);resp.setHeader("location","http://www.baidu.com");}
(2)Refresh的使用
protected void doGet(HttpServletRequest req, HttpServletResponse resp)throws ServletException, IOException {resp.setHeader("refresh","2;url=http://www.qidian.com");}
4.响应体
- 响应的数据,会将HTML,text等信息返回给客户端。
(1)响应数据传递方式
- 响应数据需要通过流来传递,response自带两个输出流
- 这两个流不能同时使用
ServletOutputStream out = response.getOutputStream();//字节流PrintWriter out = response.getWriter();//字符流
(2)示例
protected void doGet(HttpServletRequest req, HttpServletResponse resp)throws ServletException, IOException {resp.setContentType("text/html;charset=utf-8");PrintWriter out = resp.getWriter();out.print("<h1>今天天气不错</h1>");out.close();}
六.文件下载
1.创建下载文件的页面
<!DOCTYPE html>
<html lang="en">
<head><meta charset="UTF-8"><title>Title</title>
</head>
<body><a href="responseServlet">下载</a>
</body>
</html>
2.添加测试图片操作
-
去除原来的 exploded
-
重新添加exploded
3.Servlet编写代码
- 下载的源文件直接放在src目录下
protected void doGet(HttpServletRequest req, HttpServletResponse resp)throws ServletException, IOException {//告诉浏览器,以附件的形式打开数据resp.setHeader("Content-Disposition","attachment;filename=erha.png");//输入流读取文件//this.getClass().getClassLoader().getResourceAsStream("gouza.png")用于快速的获取src当中问文件对象流InputStream in = this.getClass().getClassLoader().getResourceAsStream("gouza.png");//输出流写文件ServletOutputStream out = resp.getOutputStream();//边读边写byte[] b = new byte[1024];int len = 0;while ((len = in.read(b)) != -1) {out.write(b,0,len);}out.close();in.close();}
七.请求转发与重定向
1.Servlet进行跳转的方式
跳转方式 | 方法 |
---|---|
请求转发 | request.getRequestDispatcher(“跳转的文件路径”).forward(request,response); |
重定向 | response.sendRedirect(“跳转的文件路径”); |
2.请求转发
(1)请求转发概述
- 服务器行为,客户端不知道整个操作在服务器中到底处理了多少次,所以地址栏不会发生改变
- 整个过程中只发送了一个请求,所以request对整个转发过程中所有的Servlet是共享的
- 请求转发的是那种数据提交方式,那么第二个servlet也是该方式,出form表单和超链接
(2)创建FirstServlet
package com.demo.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("/firstServlet")
public class FirstServlet extends HttpServlet {@Overrideprotected void doGet(HttpServletRequest req, HttpServletResponse resp)throws ServletException, IOException {System.out.println("进入FirstServlet");req.setAttribute("color","red");System.out.println(req.getAttribute("color"));req.getRequestDispatcher("secondServlet").forward(req,resp);}
}
(3)创建SecondServlet
package com.demo.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("/secondServlet")
public class SecondServlet extends HttpServlet {@Overrideprotected void doGet(HttpServletRequest req, HttpServletResponse resp)throws ServletException, IOException {System.out.println("进入SecondServlet");System.out.println(req.getAttribute("color"));}
}
(4)index.html,点击测试跳转
<!DOCTYPE html>
<html lang="en">
<head><meta charset="UTF-8"><title>Title</title>
</head>
<body><a href="firstServlet">测试请求转发</a>
</body>
</html>
(5)结果
3.重定向
(1)重定向概述
- 客户端行为,客户端向FirstServlet发送请求,FirstServlet给了客户端响应告诉客户端去找SecondServlet,客户端向SecondServlet发送请求
- 地址栏会变,request不共享
(2)创建FirstServlet
package com.demo.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("/firstServlet")
public class FirstServlet extends HttpServlet {@Overrideprotected void doGet(HttpServletRequest req, HttpServletResponse resp)throws ServletException, IOException {System.out.println("进入FirstServlet");req.setAttribute("color","red");System.out.println(req.getAttribute("color"));resp.sendRedirect("secondServlet");}
}
(3)创建SecondServlet
package com.demo.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("/secondServlet")
public class SecondServlet extends HttpServlet {@Overrideprotected void doGet(HttpServletRequest req, HttpServletResponse resp)throws ServletException, IOException {System.out.println("进入SecondServlet");System.out.println(req.getAttribute("color"));}
}
(4)测试
4.请求转发与重定向的区别
请求转发 | 重定向 | |
---|---|---|
发出位置 | 服务器行为 | 客户端行为 |
地址栏内容 | 地址栏不会发生改变 | 地址栏会发生改变 |
请求次数 | 只有一次请求 | 至少两次请求 |
request共享 | equest是共享的 | request不共享的 |
转发效率 | 高 | 较低 |
发送范围 | 只能在当前模块中进行 | 可以跳转到其他模块的作用域容器对象:作用域里面的作为容器的对象 |
--------------------------------------------------------: |
| 发出位置 | 服务器行为 | 客户端行为 |
| 地址栏内容 | 地址栏不会发生改变 | 地址栏会发生改变 |
| 请求次数 | 只有一次请求 | 至少两次请求 |
| request共享 | equest是共享的 | request不共享的 |
| 转发效率 | 高 | 较低 |
| 发送范围 | 只能在当前模块中进行 | 可以跳转到其他模块的作用域容器对象:作用域里面的作为容器的对象 |
相关文章:

Servlet详解
一.Servlet生命周期 初始化提供服务销毁 1.测试生命周期 package com.demo.servlet;import javax.servlet.*; import java.io.IOException;public class LifeServlet implements Servlet {Overridepublic void init(ServletConfig servletConfig) throws ServletException {…...

遥遥领先,免费开源的django4-vue3前后端分离项目
星域后台管理系统前端介绍 🌿项目简介 本项目前端基于当下流行且常用的vue3作为主要技术栈进行开发,融合了typescript和element-plus-ui,提供暗黑模式和白昼模式两种主题以及全屏切换,开发bug少,简单易学,…...

行业安卓主板-基于RK3568/3288/3588的AI智能网络广告机/自动售货机/收银机解决方案(三)
广告机 智能网络广告机通过网络将音视频、图片、文档、网页等自由排版创建成节目发布到终端。可针对不同的终端统一管理,统一发布;针对应用场景的集中和分散,可以选用局域网管理和云服务器管理。 自动售货机 随着物联网、大数据、人工智能的…...
寻找二维数组的最大值和对应下标 | C语言代码
题目: 本题目要求读入M(最大为10)行N(最大为15)列个元素,找出其中最大的元素,并输出其行列值。 输入格式: 输入在第一行中给出行数m和列数n。接下来输入m*n个整数。 输出格式: 输出最大值的行号,列号,值。 输入样例…...
2311dC++连接与串
原文 extern(C)函数使用在装饰名中包括参数类型的C装饰名.但是,因为C没有像D的T[]内置切片类型,因此C没有有效的D切片装饰. 因此,无法编译以D切片为参数的extern(C)函数. 为此,可按结构转换切片: struct DSlice(T) {T* ptr;size_t length;T[] opIndex() > ptr[0 .. length]…...

macOS 下 starUML 软件激活方案
starUML每次打开都弹出提示其实挺烦的,于是研究了一下如何 po 解(激活)它。记录一下方法以便以后使用。 我觉得这个软件很好用,大型项目的所有图我都是用这个软件画的。 直接上步骤!先关掉starUML 1、安装 asar,以便可以打开 asa…...

一文读懂从 CPU 多级缓存 缓存一致性协议(MESI)到 Java 内存模型
文章目录 CPU 多级缓存 & 缓存一致性协议(MESI)CPU 多级缓存缓存一致性协议(MESI)缓存行(Cache line)四种缓存状态缓存行状态转换多核协同示例网站体验 MESI优化和引入的问题Store Bufferes & Inva…...

MongoDB设置密码
关于为什么要设置密码 公司的测试服务器MongoDB服务对外网开放的,结果这几天发现数据库被每天晚上被人清空的了,还新建了个数据库,说是要支付比特币。查了日志看到有个境外的IP登录且删除了所有的集合。所以为了安全起见,我们给m…...

重生奇迹mu召唤师怎么加点?
召唤师在重生奇迹mu游戏里面是一个智力型的职业,所以智力自然就成为主要加点属性,但是此职业却又算是近身攻击,因为她的技能范围并不算远,而且还是呈现出一种半径趋势,一方面是攻击伤害,另一方面则是辅助造…...

第九章《搞懂算法:决策树是怎么回事》笔记
决策树算法是机器学习中很经典的一个算法,它既可以作为分类算法,也可以作为回归算法。 9.1 典型的决策树是什么样的 决策树算法是依据“分而治之”的思想,每次根据某属性的值对样本进行分类,然后传递给下个属性继续进行分类判断…...

jar包的精细化运营,Java模块化简介 | 京东云技术团队
图:模块化手机概念 一、什么是Java模块化 Java模块化(module)是Java9及以后版本引入的新特性。 官方对模块的定义为:一个被命名的,代码和数据的自描述集合。( the module, which is a named, self-descri…...

「Verilog学习笔记」移位运算与乘法
专栏前言 本专栏的内容主要是记录本人学习Verilog过程中的一些知识点,刷题网站用的是牛客网 分析 1、在硬件中进行乘除法运算是比较消耗资源的一种方法,想要在不影响延迟并尽量减少资源消耗,必须从硬件的特点上进行设计。根据寄存器的原理&a…...

静态、友好、内在:解析C++中的这些特殊元素和对象复制的优化
W...Y的主页 😊 代码仓库分享💕 🍔前言: 前面我们学习了C中关于类与对象的许多知识点,今天我们继续学习类与对象,最后再总结一下类与对象中的一些关键字内容,以及需要注意的细节。满满的干货…...

【RabbitMQ】 RabbitMQ 消息的延迟 —— 深入探索 RabbitMQ 的死信交换机,消息的 TTL 以及延迟队列
文章目录 一、死信交换机1.1 什么是死信和死信交换机1.2 死信交换机和死信队列的创建方式 二、消息的 TTL2.1 什么是消息的 TTL2.2 基于死信交换机和 TTL 实现消息的延迟 三、基于 DelayExchang 插件实现延迟队列3.1 安装 DelayExchang 插件3.2 DelayExchang 实现消息延迟的原理…...

CVE-2023-34040 Kafka 反序列化RCE
漏洞描述 Spring Kafka 是 Spring Framework 生态系统中的一个模块,用于简化在 Spring 应用程序中集成 Apache Kafka 的过程,记录 (record) 指 Kafka 消息中的一条记录。 受影响版本中默认未对记录配置 ErrorHandlingDeserializer,当用户将容…...
全局变量和局部变量在for循环的使用
imageloc字典作为全局变量,然后添加到全局的列表中,每次for循环都会将最新的元素改变之前for循环添加的元素。而imageloc字典作为局部变量,则不会影响。 import numpy as np originaljson [{"joints_vis": [1,1,1,1,1,1,1,1,1,1,…...
pytorch collate_fn测试用例
collate_fn 函数用于处理数据加载器(DataLoader)中的一批数据。在PyTorch中使用 DataLoader 时,通过设置collate_fn,我们可以决定如何将多个样本数据整合到一起成为一个 batch。在某些情况下,该函数需要由用户自定义以满足特定需求。 import …...

【qemu逃逸】HITB2017-babyqemu 2019数字经济-qemu
前言 由于本地环境问题,babyqemu 环境都没有起起,这里仅仅做记录,exp 可能不正确。 HITB2017-babyqemu 设备逆向 设备定位啥的就不说了,先看下实例结构体: 其中 dma_state 结构体如下: 这里看字段猜测…...
Docker Compose学习笔记
Docker Compose用来做什么? Docker Compose 是Docker官方的开源项目。 Compose is a tool for defining and running multi-container Docker applications. With Compose, you use a YAML file to configure your application’s services. Then, with a single …...
基于树 二叉树的回溯搜索算法(DPLL)
1)全称:Davis-Putnam-Logemann-Loveland 2)思想:基于树/二叉树的回溯搜索算法,主要基于两种策略。 单子句规则:如果一个CNF范式中存在单子句L(含有一个文字的子句),取L为…...

使用VSCode开发Django指南
使用VSCode开发Django指南 一、概述 Django 是一个高级 Python 框架,专为快速、安全和可扩展的 Web 开发而设计。Django 包含对 URL 路由、页面模板和数据处理的丰富支持。 本文将创建一个简单的 Django 应用,其中包含三个使用通用基本模板的页面。在此…...

盘古信息PCB行业解决方案:以全域场景重构,激活智造新未来
一、破局:PCB行业的时代之问 在数字经济蓬勃发展的浪潮中,PCB(印制电路板)作为 “电子产品之母”,其重要性愈发凸显。随着 5G、人工智能等新兴技术的加速渗透,PCB行业面临着前所未有的挑战与机遇。产品迭代…...
PHP和Node.js哪个更爽?
先说结论,rust完胜。 php:laravel,swoole,webman,最开始在苏宁的时候写了几年php,当时觉得php真的是世界上最好的语言,因为当初活在舒适圈里,不愿意跳出来,就好比当初活在…...

为什么需要建设工程项目管理?工程项目管理有哪些亮点功能?
在建筑行业,项目管理的重要性不言而喻。随着工程规模的扩大、技术复杂度的提升,传统的管理模式已经难以满足现代工程的需求。过去,许多企业依赖手工记录、口头沟通和分散的信息管理,导致效率低下、成本失控、风险频发。例如&#…...

视频字幕质量评估的大规模细粒度基准
大家读完觉得有帮助记得关注和点赞!!! 摘要 视频字幕在文本到视频生成任务中起着至关重要的作用,因为它们的质量直接影响所生成视频的语义连贯性和视觉保真度。尽管大型视觉-语言模型(VLMs)在字幕生成方面…...

Android 之 kotlin 语言学习笔记三(Kotlin-Java 互操作)
参考官方文档:https://developer.android.google.cn/kotlin/interop?hlzh-cn 一、Java(供 Kotlin 使用) 1、不得使用硬关键字 不要使用 Kotlin 的任何硬关键字作为方法的名称 或字段。允许使用 Kotlin 的软关键字、修饰符关键字和特殊标识…...

基于IDIG-GAN的小样本电机轴承故障诊断
目录 🔍 核心问题 一、IDIG-GAN模型原理 1. 整体架构 2. 核心创新点 (1) 梯度归一化(Gradient Normalization) (2) 判别器梯度间隙正则化(Discriminator Gradient Gap Regularization) (3) 自注意力机制(Self-Attention) 3. 完整损失函数 二…...

Golang——9、反射和文件操作
反射和文件操作 1、反射1.1、reflect.TypeOf()获取任意值的类型对象1.2、reflect.ValueOf()1.3、结构体反射 2、文件操作2.1、os.Open()打开文件2.2、方式一:使用Read()读取文件2.3、方式二:bufio读取文件2.4、方式三:os.ReadFile读取2.5、写…...
嵌入式常见 CPU 架构
架构类型架构厂商芯片厂商典型芯片特点与应用场景PICRISC (8/16 位)MicrochipMicrochipPIC16F877A、PIC18F4550简化指令集,单周期执行;低功耗、CIP 独立外设;用于家电、小电机控制、安防面板等嵌入式场景8051CISC (8 位)Intel(原始…...
pycharm 设置环境出错
pycharm 设置环境出错 pycharm 新建项目,设置虚拟环境,出错 pycharm 出错 Cannot open Local Failed to start [powershell.exe, -NoExit, -ExecutionPolicy, Bypass, -File, C:\Program Files\JetBrains\PyCharm 2024.1.3\plugins\terminal\shell-int…...