【Servlet】——Servlet API 详解

news2025/1/10 16:27:20

个人主页:兜里有颗棉花糖
欢迎 点赞👍 收藏✨ 留言✉ 加关注💓本文由 兜里有颗棉花糖 原创
收录于专栏【Servlet】
本专栏旨在分享学习Servlet的一点学习心得,欢迎大家在评论区交流讨论💌

目录

  • 一、HttpServlet
  • 二、HttpServletRequest
    • 核心方法
    • 代码示例1
    • 如何获取query string和body(form格式)中的数据
    • 获取body(json格式)中的数据
      • jackon依赖的导入
  • 三、HttpServletResponse
    • setStatus
    • setHeader
    • 构造重定向响应
    • 返回一个html界面

API就是一组类和方法,本文中我们学习Servlet API的三个类:HttpServletHttpServletRequestHttpServletResponse

一、HttpServlet

HttpServlet是我们编写Servlet代码用来的核心类,通过继承这个类并重写其中的方法,然后Tomcat去调用其中的代码逻辑。可以这样理解:我们这里写的代码(重写的方法)并不是我们自己手动调用手动执行的,而是将我们写地代码交给Tomcat,让Tomcat帮助我们执行调用这些代码。

核心方法如下:

在这里插入图片描述

  • Init方法:在webapp被加载的时候会调用此方法,可以使用该方法进行初始化操作。
  • destroy方法:在webapp被销毁(或Tomcat结束)的时候执行destroy方法,可以使用该方法进行收尾工作。注意:该方法不能保证一定会被调用到。

这里分为两种情况。

第一种情况:通过8005端口给Tomcat发起一个特殊的请求,然后Tomcat就关闭了(这种情况可以调用到destroy方法)。
第二种情况:直接杀死Tomcat进程,比如说通过任务资源管理器直接结束Tomcat任务进程,此时就无法执行destroy方法。

实际开发中第二种情况占多数(尤其是在Linux中),这就提醒我们不能依赖destroy方法,因为该方法不一定被调用到。

  • service方法:每次收到请求会执行service方法,处理每个请求。一般会使用doXXX方法来替代service方法

对于上述方法,浏览器只能构造Get方法,对于其它的方法浏览器并不方便构造。如果想要构造其它的请求方法,我们可以使用ajax或者postman。

在这里插入图片描述
实际开发中用到的最多的是Get和Post。

二、HttpServletRequest

HttpServletRequestJava Servlet API提供的接口之一,用于表示客户端的HTTP请求。

一个HTTP请求中都有哪些信息都会在HttpServletRequest类中进行体现。当Tomcat接收到Http请求时,它将解析请求并将相关的信息填充到HttpServletRequest对象中,从而使开发人员可以方便地获取和处理这些信息。

核心方法

下图是HttpServletRequest的方法:
在这里插入图片描述

  • URI和URL:URI是唯一资源标识符,URL是唯一资源定位符;URI用来区分不同的资源,而URL就是用来区分不同资源的一种方式。我们可以这样理解,URL是URI的实现方式。实际开发中一般会混着用。

Enumeration getParameterNames()String getParameter(String name)
我们直到请求中可以通过一些方式把自定义数据传递到服务器中,有两种方式:
第一种方式:query stringquery string是键值对结构的数据,Tomcat收到请求后就会把query string解析成Map这样的键值对。然后使用getParameter就可以根据key获取到value;而getParameterNames是拿到所有的key。
第二种方式:body:如果是通过post form表单的形式提交表单的话,此时body中也是键值对的形式(和query string一样)。

getParameterValues方法:即一个key涉及到多个value。

Enumeration getHeaderNames()String getHeader(String name):获取到请求头中的键值对,Tomcat收到请求后也会把请求头解析成Map。getHeader是通过key拿到指定的value;而getHeaderNames是拿到所有的key。

代码示例1

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.util.Enumeration;

@WebServlet("/request")
public class RequestServlet extends HttpServlet {
    @Override
    protected void doGet(HttpServletRequest req, HttpServletResponse resp) throws ServletException, IOException {
        // 下行代码就是为了显示告诉浏览器,浏览器拿到的内容是html
        resp.setContentType("text/html");

        // 调用req的各个方法,把得到的结果汇总到一个字符串中,然后统一返回到桌面上
        StringBuilder respBody = new StringBuilder();

        // 由于下面的内容在浏览器是按照html展示的,\n在浏览器上并不是换行,所以使用<br>来表示换行
        // 获得HTTP请求协议的版本号
        respBody.append(req.getProtocol());
        respBody.append("<br");
        // 获取HTTP请求方法
        respBody.append(req.getMethod());
        respBody.append("<br>");
        // 获取请求的URI
        respBody.append(req.getRequestURI());
        respBody.append("<br>");
        // 将req.getContextPath()返回的上下文路径添加到respBody中
        respBody.append(req.getContextPath());
        respBody.append("<br>");
        // 返回的查询字符串添加到respBody字符串中的代码
        respBody.append(req.getQueryString());
        respBody.append("<br>");

        // 拼接header
        Enumeration<String> headers = req.getHeaderNames();
        while(headers.hasMoreElements()) {
            String header = headers.nextElement();
            respBody.append("<br>");
            respBody.append(header + ": " + req.getHeader(header));
        }
        resp.getWriter().write(respBody.toString());
    }
}

跑起来之后如下图:
在这里插入图片描述

如何获取query string和body(form格式)中的数据

如何获取query string中的数据

代码如下:

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("/parameter")
public class ParameterServlet extends HttpServlet {
    @Override
    protected void doGet(HttpServletRequest req, HttpServletResponse resp) throws ServletException, IOException {
        String username = req.getParameter("username");
        String password = req.getParameter("password");
        System.out.println("username=" + username);
        System.out.println("password=" + password);
        resp.getWriter().write("OK");
    }
}

运行结果如下:
在这里插入图片描述
在这里插入图片描述

如何获取中body的数据

由于body有很多种格式,所以这里就拿form表单格式(和query string格式一样是键值对的格式)来举例。这里我们让客户端发送一个Post请求,同时使用form格式的数据在body中把数据进行传递。

代码如下:

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("/parameter2")
public class Parameter2Servlet extends HttpServlet {
    @Override
    protected void doPost(HttpServletRequest req, HttpServletResponse resp) throws ServletException, IOException {
        String username = req.getParameter("username");
        String password = req.getParameter("password");
        System.out.println("username=" + username);
        System.out.println("password=" + password);
        resp.getWriter().write("OK");
    }
}

结果如下:
在这里插入图片描述

在这里插入图片描述

总结:使用getParameter()方法可以用于获取query string中的value值以及form表单中的value值。
所以我们完全可以使用一种代码来获取query string中的value值以及form表单中的value值(doPost()方法用于处理POST请求,获取表单中的参数值。doGet()方法用于处理GET请求,获取查询字符串中的参数值)。
注意:上述两份代码的servlet path是不能重复的,否则会出现404报错。

获取body(json格式)中的数据

再来回顾一下json格式:

{
	username:lisi;
	password:9521;
}

json格式虽然在开发中也是经常会用到的一种格式,但是Servlet自身不能够对json格式的数据进行解析。解决方式是引入第三方库来进行解析(将键值对还原成Map这种key value的形式)。

解析json格式的第三方库也是有很多的,我们这里使用jackon来对json进行解析。

jackon依赖的导入

在这里插入图片描述
然后这里选择一个版本(建议选旧版本)
在这里插入图片描述
然后复制粘贴到pom.xml文件中去。
在这里插入图片描述
注意是复制粘贴到<dependencies></dependencies>标签中(一个dependencies标签可以包含多个依赖)。
在这里插入图片描述
最后刷新一下即可。

jackon依赖引入的是一个类两个方法:类是ObjectMapper,用于将Java对象转换为JSON格式的字符串,也可以将JSON字符串转换回相应的Java对象。

现在我们回归正题:获取body(json格式)中的数据(即客户端中的body按照json格式进行传输)。

在这里插入图片描述
readValue方法选择上图中的版本。该方法的作用就是把json格式的字符串解析成Java对象。
第一个参数(InputStream src)可以看到是一个流对象,表示json从哪里来。
第二个参数(JavaType valueType)用于指定一个类型:即json格式的字符串需要解析成什么类型的Java对象(这是一个将json格式的字符串映射成java对象的过程)。
关于第二个参数:我们需要定义一个类,该类的属性名称与 JSON 字符串中的字段匹配(如下图)
在这里插入图片描述
完成代码如下:

import com.fasterxml.jackson.databind.ObjectMapper;

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;

//{
//    "username":"lisi",
//    "password":"465"
//}

class User {
    public String username;
    public String password;
}


@WebServlet("/json")
public class JsonServlet extends HttpServlet {
    @Override
    protected void doPost(HttpServletRequest req, HttpServletResponse resp) throws ServletException, IOException {
        ObjectMapper objectMapper = new ObjectMapper();
        User user = objectMapper.readValue(req.getInputStream(),User.class);
        System.out.println("username=" + user.username + ", password=" + user.password);
        resp.getWriter().write("OK");
    }
}

运行结果如下:
在这里插入图片描述

在这里插入图片描述

总结:readValue方法将req请求中的body(json格式)中的字符串读取并解析,然后构造成了user对象,而user中的属性就是body(json格式)中所体现的内容。

jackon还提供了一个方法可以将java对象转换成json格式的字符串。该方法为writeValueAsString()
在这里插入图片描述

三、HttpServletResponse

HttpServletResponse是用于构造HTTP响应的对象,提供了一系列方法用于操作响应的状态码、头部信息(header)和响应体数据(body)。它和HTTP响应数据相匹配,可以用于向客户端发送响应数据。然后针对状态码、header、body这些属性就可以进行设置。

注意区分下面两个概念:
请求对象:拿到请求对象之后是为了获取对象中的属性(读操作)。
响应对象:拿到响应对象之后是为了设置对象中的属性(写操作)。

而对于doXXX这样的方法的作用就是根据请求计算响应。
再强调一下:请求对象是Tomcat收到后对Http协议解析得到的对象。
而响应对象是Tomcat创建空的对象,然后在代码中把响应对象的属性设置好。

下面来看HttpServletResponse中的核心方法,如下:
在这里插入图片描述

  • addHeader():可能会出现key相同的两个键值对(一个key可以对应多个value)。另外,Java标准库的Map是不允许key重复的,Tomcat内部也不一定把header解析成Map。
  • setHeader():一个key对应一个value。
  • setCharacterEncoding:告诉浏览器按照什么样的字符集来解析响应的body。
  • sendRedirect:用来设定重定向响应。

setStatus

代码示例1如下:
在这里插入图片描述
运行结果如下:
在这里插入图片描述

代码示例2如下:
在这里插入图片描述
运行结果如下:
在这里插入图片描述

在返回状态码的同时,可以给body写入数据,这样就可以得到比较个性化的错误页面,比如:
在这里插入图片描述
上图就是搜狗的404错误页面。
在这里插入图片描述
上图就是百度的404错误页面。
在这里插入图片描述
上图是哔哩哔哩的404错误页面。

Tomcat同样有一个内置的404报错页面,虽然setStatus方法并没有页面,但是sendError方法是有页面的。代码如下:
在这里插入图片描述
在这里插入图片描述
在这里插入图片描述
上图是Tomcat的404报错页面。

setHeader

我们通过使用setHeader方法来设置任意的响应报头。举例:通过refresh属性来设置浏览器自动刷新(比如refresh:2,意思就是浏览器每隔两秒就会自动刷新)。

代码如下:
在这里插入图片描述

运行结果如下(下图中的时间每隔两秒就会改变):在这里插入图片描述
在这里插入图片描述
在这里插入图片描述

构造重定向响应

让页面重定向到百度主页。

注意:使用重定向一定要带有Location属性,然后用这个属性来描述你要重定向到哪个地方

代码如下:
在这里插入图片描述

结果如下:
在这里插入图片描述

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

补充:
301:永久重定向
302:临时重定向

刚刚上述代码可以等价替换成下面代码:
在这里插入图片描述在这里插入图片描述

resp.sendRedirect("https://www.baidu.com");等价于resp.setStatus(302);resp.setHeader("Location","https://www.baidu.com");(浏览器看到302和https://www.baidu.com这两个字段就知道要跳转到百度)

返回一个html界面

在这里插入图片描述
结果如下:
在这里插入图片描述
为什么会出现乱码呢?

我们要知道,如果在IDEA中直接写一个中文字符串的话是按照utf-8进行编码的。但是浏览器默认是使用操作系统的编码方式来解析和显示页面,而windows简体中文版的默认编码方式是gdk,此时浏览器按照gdk的方式来解析utf-8的话就会出现上图中的乱码。

所以我们要设置一下让浏览器按照utf-8的方式进行解析(如下)。
在这里插入图片描述
在这里插入图片描述

上述代码还存在一个问题,如下(将doGet方法中的两行代码进行互换):
在这里插入图片描述
在这里插入图片描述
可以看到又出现了乱码。在Servlet中,为resp设置属性的时候,需要注意顺序:要先设置header,然后再设置body;否则如果先设置body的话此时header、status就已经定性而来不及修改了

好了,以上就是本文的全部内容了。希望各位友友可以一键三连哈!!!

在这里插入图片描述

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

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

相关文章

【unity小技巧】unity3d环境带雾的昼夜系统变化

最终效果 文章目录 最终效果眩光素材眩光配置全局灯光配置天空盒配置天空盒资产配置天空盒&#xff0c;开启雾 代码控制天空盒 环境 雾 灯光昼夜交替变化参考完结 眩光素材 链接&#xff1a;https://pan.baidu.com/s/1qlFSJSju6ZjwCylwkh14eA?pwdveww 提取码&#xff1a;veww…

Oracle和Mysql数据库

数据库 Oracle 体系结构与基本概念体系结构基本概念表空间(users)和数据文件段、区、块Oracle数据库的基本元素 Oracle数据库启动和关闭Oracle数据库启动Oracle数据库关闭 Sqlplussqlplus 登录数据库管理系统使用sqlplus登录Oracle数据库远程登录解锁用户修改用户密码查看当前语…

Android学习之路(29) Gradle初探

前言: 大家回想一下自己第一次接触Gradle是什么时候&#xff1f; 相信大家也都是和我一样&#xff0c;在我们打开第一个AS项目的时候&#xff0c; 发现有很多带gradle字样的文件&#xff1a;setting.gradle, build.gradle,gradle.warpper,以及在gradle文件中各种配置&#xff…

来看看Tomcat和Web应用的目录结构

在前面两篇大致了解了Tomcat的架构和运行流程&#xff0c;以及Tomcat应用中的web.xml。 聊一聊Tomcat的架构和运行流程&#xff0c;尽量通俗易懂一点-CSDN博客 来吧&#xff0c;好好理解一下Tomcat下的web.xml-CSDN博客 那接下来&#xff0c;再看看Tomcat的目录&#xff0c;…

BeanDefinitionRegistry学习

Spring版本5.1.x 简介 在Spring框架中&#xff0c;BeanDefinitionRegistry是一个接口&#xff0c;它主要用于向注册表中注册BeanDefinition实例&#xff0c;完成注册的过程。该接口的主要方法是registerBeanDefinition&#xff0c;用于将一个BeanDefinition实例注册到注册表中…

分享一个WPF项目

最近在学习WPF开发方式&#xff0c;找到一些项目进行拆解学习&#xff1b;本位主要分享一个WPF项目&#xff0c;叫做WPFDevelopers&#xff0c;在git上大约有1.3K星&#xff0c;话不多说&#xff0c;先看看效果&#xff1a; 这个项目开发可以编译启动后直接查看样例、Xaml、Cha…

如何搭建私有云盘SeaFile并实现远程访问本地文件资料

&#x1f308;个人主页: Aileen_0v0 &#x1f525;热门专栏: 华为鸿蒙系统学习|计算机网络|数据结构与算法 ​&#x1f4ab;个人格言:“没有罗马,那就自己创造罗马~” #mermaid-svg-hsDnDEybLME85dTx {font-family:"trebuchet ms",verdana,arial,sans-serif;font-siz…

让AI作为人类的“副驾驶”

让AI作为副驾驶&#xff0c;辅助人类完成工作&#xff0c;是人工智能技术在实际应用中的一个重要方向。这个设想在多个领域都有广泛应用的可能性&#xff1a; 驾驶场景&#xff1a;自动驾驶技术中&#xff0c;AI副驾驶可以实时监测路况、规划最优行驶路线、预警潜在危险&#x…

接口自动化测试详解

&#x1f345; 视频学习&#xff1a;文末有免费的配套视频可观看 &#x1f345; 关注公众号【互联网杂货铺】&#xff0c;回复 1 &#xff0c;免费获取软件测试全套资料&#xff0c;资料在手&#xff0c;涨薪更快 引言 与UI相比&#xff0c;接口一旦研发完成&#xff0c;通常变…

Unity | Spine动画记录

https://blog.csdn.net/linshuhe1/article/details/79792432 https://blog.csdn.net/winds_tide/article/details/128925407 1.需要的三个文件 通常制作好的 Spine 动画导出时会有三个文件&#xff1a; .png 、.json 和 .atlas&#xff1a; skeleton-name.json 或 skeleton-…

【10秒开服】雾锁王国服务器全自动部署教程

你是火焰之子&#xff0c;一个濒死种族最后的希望火苗。苏醒吧&#xff0c;克服腐化一切的迷雾所裹挟的恐怖&#xff0c;重新夺回你的王国所失落的瑰丽。置身于广袤世界&#xff0c;战胜难以想象的强大Boss&#xff0c;修造宏伟厅堂&#xff0c;在这款至多16名玩家的合作类生存…

LF will be replaced by CRLF the next time Git touches it

参考这篇文章git提示“warning: LF will be replaced by CRLF”的解决办法-CSDN博客

问题:魁奈在税收政策方面的主张是() #学习方法#其他#经验分享

问题&#xff1a;魁奈在税收政策方面的主张是&#xff08;&#xff09; A.征收农业税 B.征收工商业税 C.征收间接税 D.征收地租税 参考答案如图所示

挑战杯 opencv 图像识别 指纹识别 - python

0 前言 &#x1f525; 优质竞赛项目系列&#xff0c;今天要分享的是 &#x1f6a9; 基于机器视觉的指纹识别系统 &#x1f947;学长这里给一个题目综合评分(每项满分5分) 难度系数&#xff1a;3分工作量&#xff1a;3分创新点&#xff1a;4分 该项目较为新颖&#xff0c;适…

奇瑞瑞虎8,是真不能随便碰

文 | AUTO芯球 作者 | 李虎 我是实在看不下去了啊 这奔驰车主砸车 现在开始反转了啊 但卡住我喉咙的是定损5200的奇瑞引擎盖啊 我是真买不起&#xff0c;也不敢买啊 我怕A柱断了&#xff0c;要修20万啊 但我一算&#xff0c;这也不对啊 顶配的报价也只有16.18万啊 如果…

TCP TIME_WAIT 过多怎么处理

文章目录 1.什么是 TCP TIME_WAIT&#xff1f;2.为什么要 TIME_WAIT?3.TIME_WAIT 过多的影响4.解决办法4.1 调整短连接为长连接4.2 调整系统内核参数 5.小结参考文献 1.什么是 TCP TIME_WAIT&#xff1f; TCP 断开连接四次挥手过程中&#xff0c;主动断开连接的一方&#xff…

计算机毕业设计SSM基于的流浪宠物救助系统

项目运行 环境配置&#xff1a; Jdk1.8 Tomcat7.0 Mysql HBuilderX&#xff08;Webstorm也行&#xff09; Eclispe&#xff08;IntelliJ IDEA,Eclispe,MyEclispe,Sts都支持&#xff09;。 项目技术&#xff1a; vue mybatis Maven mysql5.7或8.0等等组成&#xff0c;B…

嵌入式中VMware 三种网络模式的详解与配置方法

今天梳理下 VMware虚拟机的三种网络工作模式。 相信通过这两篇文章&#xff0c;能够帮助你全面搞懂虚拟机的网络工作模式和配置。 VMware的三种网络工作模式 vmware为我们提供了三种网络工作模式, 它们分别是: Bridged&#xff08;桥接模式&#xff09;、NAT&#xff08;网…

Pytest测试用例参数化

pytest.mark.parametrize(参数名1,参数名2...参数n, [(参数名1_data1,参数名2_data1...参数名n_data1),(参数名1_data2,参数名2_data2...参数名n_data2)]) 场景&#xff1a; 定义一个登录函数test_login,传入参数为name,password&#xff0c;需要用多个账号去测试登录功能 # …

八. 实战:CUDA-BEVFusion部署分析-学习spconv的优化方案(Implicit GEMM conv)

目录 前言0. 简述1. 什么是Implicit GEMM Conv2. Explicit GEMM Conv3. Implicit GEMM Conv4. Implicit GEMM Conv优化5. spconv和Implicit GEMM Conv总结下载链接参考 前言 自动驾驶之心推出的 《CUDA与TensorRT部署实战课程》&#xff0c;链接。记录下个人学习笔记&#xff0…