网站开发--详解Servlet

💕"Echo"💕
作者:Mylvzi
文章主要内容:网站开发–详解Servlet
在这里插入图片描述

一.基本介绍

tomcat是Java中开发服务器的重要的一个工具,任何开发的服务器都要部署在tomcat之上,可以说tomcat是所有服务器的底座,为了更好的操作http,tomcat对原生的api进行了封装,封装为Servlet*,通过servlet就可以非常方便的完成和http协议的祥光操作

关于Servlet的学习,主要学习以下三个类即可:

  1. HttpServlet
  2. HttpRequestServlet
  3. HttpResponseServlet

掌握好这三个类,就能很好的使用Servlet

二.HttpServlet

HttpServlet是Servlet中最核心的一个类,我们创建的用于进行交互的类都必须要继承于HttpServlet

在这里插入图片描述

HttpServlet中常用方法
在这里插入图片描述

  1. init()方法会在一个Servlet对象被创建好之后,自动的调用这个方法,完成初始化操作
  2. destroy()方法会在Servlet对象被销毁之前进行调用,完成一些释放资源的工作
  3. service()方法实际上并不会直接使用到,而是被doGet(),doPost()给替代

init(),destroy(),service()这三个方法在实际的开发中很少用到,都是tomcat自动的在合适的时机帮助我们调用的,这三个方法经常会出现到一个经典的面试题中:

经典面试题:Servlet的生命周期

Servlet的生命周期是指Servlet实例被创建出来到被销毁的过程,具体来说可以分为以下几点:

  1. Servlet实例的创建:当客户端发送请求,服务器就要创建出一个Servlet实例来处理请求,通过构造方法创建出一个Servlet实例
  2. 初始化:在实例被创建好之后,会自动调用init()方法,来完成一些初始化操作,比如:读取配置文件,数据库的连接,一些资源的初始化
  3. 处理请求:当有客户端的http请求发送到服务器时,Servlet会先调用service()方法,判断请求的具体类型(根据请求的方法),service()方法就像是火车的中转站一样,看你的目的是什么,再给你对应的路线.这里也是一样,如果请求是GET方法,就会交给代码中的doGet()方法处理,如果是POST请求,就会交给doPost()方法
  4. 实例的销毁:当我们不再使用Servlet实例时(客户端不再有请求发送过来/人为关闭),先调用destroy()方法,做一些清理工作,比如数据库的断开,资源释放等

这个过程类似于生命周期中的起始、成长、运行和结束阶段,每个阶段都有其特定的任务和目的。

总的来说,我们所写的代码都要继承于HttpServlet这个类,重写里面的方法,并将重写的方法插入到tomcat已有的框架之中,我们只需关注数据处理部分,其余操作都是tomcat自动帮助我们完成

二.HttpServletRequest

1.介绍

HttpServletRequest表示一个Http请求类,用于接受从客户端传输过来的http请求

HttpServletRequest中的很多属性和请求头的相关属性相同,理解好请求头中的相关属性有助于学习HttpServletRequest类

一个Http请求由四部分组成:

  1. 首行 = 方法 + URL + 版本号
  2. 请求头:由键值对组成
  3. 空行
  4. body

2.HttpServletRequest常用方法介绍

在这里插入图片描述

在这里插入图片描述

1.获取首行信息和header中的数据**

@WebServlet("/request1")
public class RequestParameter extends HttpServlet {@Overrideprotected void doGet(HttpServletRequest req, HttpServletResponse resp) throws ServletException, IOException {// 尝试获取请求报文中的详细信息StringBuffer stringBuffer = new StringBuffer();// 这个是线程安全的// 1.获取首行信息stringBuffer.append(req.getMethod());stringBuffer.append("<br>");// <br>是html中的换行标签stringBuffer.append(req.getRequestURI());stringBuffer.append("<br>");stringBuffer.append(req.getProtocol());stringBuffer.append("<br>");// 2.获取请求头信息Enumeration<String> enumeration = req.getHeaderNames();while (enumeration.hasMoreElements()){// 利用迭代器访问每一个头部keyString key = enumeration.nextElement();String val = req.getHeader(key);// 获取key的值stringBuffer.append(key + ":" + val + "<br>");}resp.getWriter().write(stringBuffer.toString());}
}

获取结果:
在这里插入图片描述

总结:
对于请求报文来说,获取首行中的信息(版本号/方法/URL),可以直接通过指定的get方法来获取,获取请求头(header)中内容,可以利用迭代器,将请求头中的所有键值对都存储到一个集合之中,再利用getHeader方法,指定参数,来获取对应的value值

2.获取请求报文中的query string**

query string,译为查询字符串,往往和用户的关键信息相关(登录密码),在日常的开发中,我们也经常利用到query string来完成一些业务上的逻辑,所以获取请求报文中的query string是一个很频繁且重要的操作

query string在请求报文中的位置是不固定的,具体位置取决于请求报文中的方法(method),如果方法是Get,query string就位于URL之中,如果方法是Post,query string就位于body之中

注意:实际上URL中是包含query string的,但是在servlet的getURL类似的方法中返回的是一个不携带query string的URL(可能是因为查询字符串这个操作过于频繁,就单独拿出来进行包装了)

1.方法为Get

说明:前后端交互的一个关键点在于前端和后端要对传输内容的格式都了然于胸,后端要知道前端是如何组织数据进行传输的

比如,规定好客户端发送的请求报文中的query string的格式是类似于:

username=zhangsan&password=123456

那么服务器这边就可以规定好的格式进行解析

代码演示:

@WebServlet("/request2")
public class GetRequestParameter extends HttpServlet {@Overrideprotected void doGet(HttpServletRequest req, HttpServletResponse resp) throws ServletException, IOException {// 获取用户传输过来的query string// 假设用户传输过来的query string的格式是形如 username='lisi'&password='123'这样的形式// servlet会自动将query string中的所有内容存储到一个Map结构之中(键值对都存储到其中// 后端程序员只需通过getParameter()方法来获取具体的值即可// 注意你构造的query string内部不需要有引号!!!// 可以根据这个特性来完成很多操作  比如判断用户名是否正确// 在传输中文的时候最好传输urlencode之后的内容String username = req.getParameter("username");String password = req.getParameter("password");System.out.println("username=" + username);System.out.println("password=" + password);}
}

在这里插入图片描述

当在浏览器中输入:“http://127.0.0.1:8080/testsevlet/request2?username=zh&password=123”

在终端上显示:
在这里插入图片描述

如果不添加query string,显示的结果就是null

在这里插入图片描述

总结:

对于Get方法的请求,query string位于首行中,直接使用getParameter方法,并制定参数,就能获取到对应的值

2.方法为Post

此时query string位于body之中,而body的传输是有不同的格式的,对于不同格式的body,要采用不同的方法来进行解析,获取query string,常用的body格式有两种:

  1. form表单
  2. json
1.form表单格式

body通过form表单来进行数据的组织,内部是键值对,Content-Type:application/x-www-form-urlencoded

获取body中的内容和上面获取query string的方法一致

通过postman发送一个post请求
在这里插入图片描述
服务器的代码不变,打印结果为:
在这里插入图片描述

总结:如果body的格式是form表单,虽然此时query string位于body之中,但是也可以直接使用**getParameter()**方法,并指定参数来获取要查询的内容

2.json格式

如果body是通过json格式来进行组织,原生的Servlet是不支持解析json格式的数据的,此时需要进入第三方库jackson来帮助进行json格式数据解析(jackson也是spring官方推荐的用于解析JSON格式数据的第三方库)

准备:引入jackson依赖

在maven中找到jackson,选择合适的版本进行路径的复制,并在项目中的pom.xml中引入

        <!-- https://mvnrepository.com/artifact/com.fasterxml.jackson.core/jackson-databind --><dependency><groupId>com.fasterxml.jackson.core</groupId><artifactId>jackson-databind</artifactId><version>2.15.0</version></dependency>

在这里插入图片描述

在这里插入图片描述
引入好依赖之后就可以使用jackson中的api解析json格式的数据

具体的操作依赖于一个类和两个核心方法:
在这里插入图片描述
ObjectMapper(对象映射器),是JSON数据与Java对象转换的核心枢纽,在交互时,通过JSON字符串进行网络传输,处理逻辑,使用Java对象

站在服务器的角度,我们收到的数据是JSON字符串,需要将其转化为Java对象(read放哪发),来完成一些业务上的逻辑,完成后,还需要将处理好的Java对象转化为JSON字符串(write方法)

1.创建一个类 用于接收JSON字符串

// 创建一个类  用于接收JSON字符串
class Request {public String username;public String password;
}

2.交互

    // 创建出ObjectMapper实例private ObjectMapper objectMapper = new ObjectMapper();@Overrideprotected void doPost(HttpServletRequest req, HttpServletResponse resp) throws ServletException, IOException {// 1.将JSON字符串转化为Java对象Request request = objectMapper.readValue(req.getInputStream(),Request.class);// 2.获取query stringString username = request.username;String password = request.password;// 3.打印String user = "username=" + username + "<br>password=" + password;System.out.println(user);// 4.将打印的结果传输给客户端// 这个方法就是和上面的read方法相反// 这里把Java对象转化为json字符串,传输给客户端String respBody = objectMapper.writeValueAsString(user);resp.getWriter().write(respBody);}

3.发送请求并打印结果

在这里插入图片描述
在这里插入图片描述

objectMapper.readValue()内部做了很多工作,最终的结果是把JSON字符串转化为Request的Java对象,内部具体实现细节为:

  1. 根据请求报文的输入流(就是方法的第一个参数)获取到请求报文中的body部分的所有内容
  2. 按照JSON格式进行解析,并把解析出来的所有的键值对都存储到一个Map之中
  3. 通过第二个参数(反射)来明确要转化的Java对象,拿着Java对象中的属性从Map之中寻找对应的值,并把对应的值赋给赋给创建出来的Java对象

通过以上三步就完成了JSON字符串到Java对象的转化,举一个简单的例子
在这里插入图片描述

以上就是关于HttpRequestServlet的所有内容,可以观察到,这个类中大部分方法都是get方法,主要原因是因为我们是想通过这个类获取到客户端发送的请求报文,而不是去修改请求报文,所以不需要设置set方法,而在下面要介绍的HttpResponseServlet类中,更多的方法就是set方法(服务器这边要返回一个响应报文)

三.HttpResponseServlet

常用方法:
在这里插入图片描述

setStatus() 设置状态码

@WebServlet("/status")
public class StatusServlet extends HttpServlet {@Overrideprotected void doGet(HttpServletRequest req, HttpServletResponse resp) throws ServletException, IOException {
//        resp.setStatus(404);// 此时响应报文的状态码就被设置为404// 如果只传输上述的状态码  浏览器是空白界面  不会有任何信息// 我们一般看到的都是有一个"错误界面"// 可以通过resp.sendError(404,"你小子搜的是什么资源???");// 这里面就比较的灵活了// 根据一定的逻辑  来灵活的输出状态码信息}
}

当需要发送报错信息时,更推荐使用第二种方式,更加灵活
结果显示:
在这里插入图片描述

setHeader 在头部中设置一些属性

自动刷新

每间隔1s就自动刷新界面,利用header中的refresh字段

@WebServlet("/refresh")
public class RefreshServlet extends HttpServlet {@Overrideprotected void doGet(HttpServletRequest req, HttpServletResponse resp) throws ServletException, IOException {resp.setHeader("refresh","1");// 1s 后自动刷新界面resp.getWriter().write("" + System.currentTimeMillis());}
}

在这里插入图片描述
在这里插入图片描述

sendRedirect() 构造重定向

当状态码是3xx时,就代表要进行重定向,利用sendRedirect方法可以直接跳转

@WebServlet("/redirect")
public class RedirectServlet extends HttpServlet {@Overrideprotected void doGet(HttpServletRequest req, HttpServletResponse resp) throws ServletException, IOException {
//        // 一个重定向的报文包含两部分:状态码 和 要跳转的界面
//        resp.setStatus(302);// 核心是要有对应的状态码(以3开头),以及要跳转的界面(Location字段)
//        resp.setHeader("Location","https://sogou.com");// 此时就会自动跳转到搜狗界面// 上述写法比较繁琐  一般来说会使用下面这个方法来设置一个跳转的响应报文resp.sendRedirect("https://sogou.com");}
}

不仅可以通过服务器这边设置跳转的界面,还可以在前端中使用JS定时器来实现相同的效果

<!DOCTYPE html>
<html>
<head><title>页面跳转示例</title><script>// 在页面加载后,等待 3 秒后跳转到新页面window.onload = function() {setTimeout(function() {window.location.href = "https://www.sogou.com";}, 3000); // 3 秒};</script>
</head>
<body><h1>页面将在 3 秒后跳转到搜狗主页</h1>
</body>
</html>

以上就是<<网站开发–详解Servlet>>的所有内容,重点掌握三个类的api的基本用法,多多使用就能融会贯通,实现更复杂的交互!

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.mzph.cn/news/703641.shtml

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

golang学习3,golang 项目中配置gin的web框架

1.go 初始化 mod文件 go mod init gin-ranking 2.gin的crm框架 go get -u github.com/gin-gonic/gin 3.go.mod爆红解决

前后端分离Vue+node.js在线学习考试系统gqw7o

与其它应用程序相比&#xff0c;在线学习平台的设计主要面向于学校&#xff0c;旨在为管理员和学生、教师、院系提供一个在线学习平台。学生、教师、院系可以通过系统及时查看公告信息等。 在线学习平台是在Windows操作系统下的应用平台。为防止出现兼容性及稳定性问题&#xf…

B站项目-基于Pytorch的ResNet垃圾图片分类

基于Pytorch的ResNet垃圾图片分类 数据集预处理 画图片的宽高分布散点图 import osimport matplotlib.pyplot as plt import PIL.Image as Imagedef plot_resolution(dataset_root_path):image_size_list []#存放图片尺寸for root, dirs, files in os.walk(dataset_root_pa…

装配制造业的MES系统种的物料齐套技术

装配是制造企业涉及产品生产加工最为普遍的一种模式&#xff0c;包括汽车、电子、电器、电气等行业。经研究表明&#xff0c;装配在整个产品制造总成本中所占比例超过了50%&#xff0c;所占用的总生产时间比例在40%-60%&#xff0c;直接影响着产品质量和成本。装配制造非常强调…

树-王道-复试

树 1.度&#xff1a; 树中孩子节点个数&#xff0c;所有结点的度最大值为 树的度 2.有序树&#xff1a; 逻辑上看&#xff0c;树中结点的各子树从左至右是有次序的&#xff0c;不能互换。 **3.**树的根节点没有前驱&#xff0c;其他节点只有一个前驱 **4.**所有节点可有零个或…

Leetcoder Day23| 回溯part03:组合+分割

语言&#xff1a;Java/Go 39. 组合总和 给你一个 无重复元素 的整数数组 candidates 和一个目标整数 target &#xff0c;找出 candidates 中可以使数字和为目标数 target 的所有不同组合 &#xff0c;并以列表形式返回。你可以按任意顺序返回这些组合。 candidates 中的同一个…

环形光源让图像质量瞬间提升,一探究竟吧!

光源对机器视觉检测系统的性能起着重要作用,精确的光学结构设计可以提高捕获图像的质量,准确地分离目标和背景信息,不充足的光线会使捕捉到的图像无法满足需求&#xff0c;针对不同的检测对象,不同的形状光源应运而生。我们来看看最常用的LED光源之一—环形光源。 环形光源&…

vue3 使用pina

一、Vue 3 项目中集成Pina 状态管理库 要在 Vue 3 项目中使用 Pina&#xff08;Vue 3 状态管理库&#xff09;&#xff0c;您可以按照以下步骤操作&#xff1a; 1. 安装 Pina 库相应的插件&#xff1a; yarn add pinia # 或者使用 npm npm install pinia 2. 在您的 Vue 3 项…

电脑休眠之后唤不醒

现象&#xff1a;午休时间电脑休眠了&#xff0c;醒来之后发现在密码输入界面&#xff0c;但鼠标键盘没反应。按重启键或电源机重新开机&#xff0c;结果开不了机。 原因&#xff1a;1、内存条脏了&#xff0c;导致内存条读取失败 2、休眠的时候硬盘休眠了&#xff0c;导致按…

AngularJS安装版本问题

一、安装 Angular CLI 脚手架安装命令&#xff1a; npm install -g angular/cli 在安装前请确保自己安装NodeJS环境版本为V18及以上&#xff0c;否则会因node版本问题导致项目无法正常运行。 脚手架安装后&#xff0c;已提示了当前node版本必须为18.13.0或大于20.9.0版本&…

git之分支管理

一.理解分支 我们看下面这张图片&#xff1a; 在版本回退⾥&#xff0c;你已经知道&#xff0c;每次提交&#xff0c;Git都把它们串成⼀条时间线&#xff0c;这条时间线就可以理解为是⼀个分⽀。截⽌到⽬前&#xff0c;只有⼀条时间线&#xff0c;在Git⾥&#xff0c;这个分⽀…

2024年 前端JavaScript入门到精通 第四天 笔记

4.1 函数的基本使用以及封装练习 ★ 函数命名规范 4.2 函数的参数以及默认参数 函数的灵魂&#xff01;&#xff01;&#xff01; 4.3 函数封装数组求和案例 4.4 函数返回值return 4.5 函数返回值细节以及上午总结 4.6 函数返回值案例-求最大值和最 4.7 函数复习以及断点进入函…

如何在Linux搭建MinIO服务并实现无公网ip远程访问内网管理界面

文章目录 前言1. Docker 部署MinIO2. 本地访问MinIO3. Linux安装Cpolar4. 配置MinIO公网地址5. 远程访问MinIO管理界面6. 固定MinIO公网地址 前言 MinIO是一个开源的对象存储服务器&#xff0c;可以在各种环境中运行&#xff0c;例如本地、Docker容器、Kubernetes集群等。它兼…

挑战杯 基于机器视觉的12306验证码识别

文章目录 0 简介1 数据收集2 识别过程3 网络构建4 数据读取5 模型训练6 加入Dropout层7 数据增强8 迁移学习9 结果9 最后 0 简介 &#x1f525; 优质竞赛项目系列&#xff0c;今天要分享的是 基于机器视觉的12306验证码识别 该项目较为新颖&#xff0c;适合作为竞赛课题方向…

Inno setup 打包jar包+前端dist+mysql+navicat等应用文件操作

目录 一、 使用exe4j将后端jar包打包成exe应用文件 1.创建一个新的工程 2.选择一个你想要存放的路径 3.进入配置界面 4.选择jar转换exe模式 5.自定义名字和选择输出路径 6.配置初始化 7.配置java环境 8.测试运行结果 二、Inno 打包应用文件exe 1.新建一个工程文件 2…

【hashmap】【将排序之后的字符串作为哈希表的键】【获取 HashMap 中所有值的集合】Leetcode 49 字母异位词分组

【hashmap】【将排序之后的字符串作为哈希表的键】【获取 HashMap 中所有值的集合】Leetcode 49 字母异位词分组 解法1 将排序之后的字符串作为哈希表的键解法2 在解法一的基础上加入了getOrDefault ---------------&#x1f388;&#x1f388;题目链接&#x1f388;&#x1f3…

从零开始手写mmo游戏从框架到爆炸(二十二)— 战斗系统三

导航&#xff1a;从零开始手写mmo游戏从框架到爆炸&#xff08;零&#xff09;—— 导航-CSDN博客 目录 地图设定 战斗引擎 服务端的BattleHandler 客户端的相关handler 战斗场景展示 执行效果 文接上一章。我们把战斗系统demo应用到实际的项目中来。在第十九章&#xf…

AI对话系统app开源

支持对接gpt&#xff0c;阿里云&#xff0c;腾讯云 具体看截图 后端环境&#xff1a;PHP7.4MySQL5.6 软件&#xff1a;uniapp 废话不多说直接上抗揍云链接&#xff1a; https://mny.lanzout.com/iKFRY1o1zusf 部署教程请看源码内的【使用教程】文档 欢迎各位转载该帖/源码

智慧公厕是什么?智慧公厕意义何在

随着城市化进程的加速&#xff0c;公厕成为城市管理中不容忽视的一环。智慧公厕传统的公厕管理方式已经无法满足当今社会的需求&#xff0c;因此智慧公厕的出现成为解决问题的利器。什么是智慧公厕&#xff1f;智慧公厕是实现公共厕所信息化、数字化、智慧化全方位管理与服务的…

啤酒:精酿啤酒与烧烤的热烈碰撞

在夏日的傍晚&#xff0c;烧烤与啤酒总是绝配。当Fendi Club啤酒遇上烧烤&#xff0c;它们将为我们带来一场热烈的美味碰撞。 Fendi Club啤酒&#xff0c;以其醇厚的口感和淡淡的麦芽香气而著称。这款啤酒在酿造过程中采用了特别的工艺&#xff0c;使得酒体呈现出诱人的金黄色&…