Servlet(三个核心API介绍以及错误排查)【二】

server/2024/9/24 6:22:24/

文章目录

  • 一、三个核心API
    • 1.1 HttpServlet
      • 【1】地位
      • 【2】方法
    • 1.2 HttpServletRequest
      • 【1】地位
      • 【2】方法
      • 【3】关于构造请求
    • 1.3 HttpServletResponse
      • 【1】地位
      • 【2】方法
  • 四、涉及状态码的错误排查(404……)
  • 五、关于自定义数据 ---- body或query String 里面的内容
    • 5.1 地位
    • 5.2 给服务器传递自定义数据
    • 5.3 获取自定义数据

学习Servlet就是学习API以及如何进行web开发

一、三个核心API

1.1 HttpServlet

【1】地位

编写 Servlet 代码会用到的核心的类。我们通过继承这个类并重写其中的方法,把代码交给Tomcat,Tomcat负责在合适的时机去调用

【2】方法

  1. init()
    (1)使用时机:原则上 init() 是在Tomcat启动加载webapp的时候就执行了,但是Tomcat是可以配置webapp为“懒加载”状态的(让webapp在首次且真正被访问到的情况下才加载)

    (2)作用:进行一些初始化操作

  2. destory()
    (1)使用时机:webapp 被销毁的时候执行

    (2)作用:进行一些收尾工作

    (3)该方法不一定会被调用到:因为destory()是webapp被销毁时被调用到,而Tomcat有两种方式结束。实际开发中,以第二种方式居多,所以我们不能太依赖destory()

    • 一种是通过 8005 端口,给Tomcat发送特殊的请求(告诉Tomcat要关闭了),然后Tomcat调用destory()
    • 直接杀死Tomcat进程,如任务管理器上直接结束进程,此时Tomcat没有机会执行destory()
  3. service()
    (1)使用时机:每次收到请求,都会处理service。实际开发中用得不多,多用下面的do三部曲。

    (2)作用:处理每个请求

  4. doGet()
    (1)使用时机:收到Get请求时

    (2)作用:处理Get请求的情景

  5. doPost()
    (1)使用时机:收到 Post 请求时

    (2)作用:处理 Post 请求的情景

  6. doPut/doDelete……:使用时机同doGet、doPost,遇到对应的请求会被调用

1.2 HttpServletRequest

【1】地位

Tomcat 收到HTTP请求后,就会把请求解析成该对象

【2】方法

  1. getProtocol():返回请求协议的名称和版本
  2. getMethod():回请求的 HTTP 方法的名称,例如,GET、POST 或 PUT
  3. getRequestURI():返回该请求的 URL
    • 什么是URI:URL可以理解为是URI的一种实现方式,但是两者经常混用
      • URI指的是唯一资源标识符,可以区分不同的资源
      • URL则是唯一资源定位符,用来描述网络上一个资源,又因为资源是唯一的,所以可以定位资源
  4. 获取请求中参数的值
    • getParameterNames():拿到所有的key
    • getParameter(String name):根据key拿到value
    • getParameterValues(String name):涉及到一个key对应到多个value值的情况,并不经常使用
  5. 获取请求头中的内容:获取到请求头的键值对。Tomcat 在收到请求之后会把请求头解析成Map。
    • getHeaderNames()
    • getHeader(String name)
  6. 获取请求主体中的内容
    • getCharacterEncoding():获取到 contentText 里的字符集。字符集本质上是 contentType 的一部分,但是由于关于字符集的操作使用很频繁,所以被单独拉出来
    • getContentType():获取到 body 里的类型
    • getContentLength():获取到 body 的长度
    • getInputStream():获取到能读取到body内容的流对象

【3】关于构造请求

  1. 服务器只能比较方便地构造出Get请求,至于其他类型的请求,我们需要使用ajax或者postman构造
  2. 实际开发中,常用的请求是 Get或Post

1.3 HttpServletResponse

【1】地位

与 HTTP响应数据匹配

【2】方法

  1. 给 header 赋值
    • setHeader(String name, String value):如果存在,新覆盖旧
    • addHeader(String name, String value):如果存在,新旧并存
      • 一般来说,约定键值对的key是要唯一的,但是实践中,可能会出现一个key对应多个value的情况,但这个是由浏览器/HTTP客户端来控制的
      • Map无法允许key重复矛盾:注意,【Tomcat会将header解析成Map形式】只是一种粗略说法,实际上Tomcat内部不一定是把header解析成Java标准库的Map。Java标准库的Map并不允许key重复,但有第三方库是允许的
    • 示例:通过 refresh 属性,设置浏览器自动刷新
      • resp.setHeader(“Refresh”, 2): 浏览器每隔2s自动刷新一次。
        实际上并非是精确的2000ms,一般会大一些。因为从 “浏览器发起请求” 到 “服务器响应”再到“页面被解析出来”都是需要消耗一定的时间的
  2. 设置状态码
    • setStatus(int sc)
    • 关于个性化错误页面:Tomcat 是可以在返回状态码的时候,给body写入数据的,此时就可以得到一个“个性化的错误页面”。Tomcat有自己内置的错误页面,比如【resp.sendError(404)】,但由于太丑,日常开发中我们往往还是会选择自定义

四、涉及状态码的错误排查(404……)

  1. 404:浏览器要访问的资源,服务器上并不存在
    • 检查请求路径和服务器配置是否一致
    • 确认webapp是否被正确加载:我们的程序是通过
  2. 405:方法不允许
    • 请求发起的方法与对应的doXX方法不匹配。如在浏览器中输入一个url地址,就是发起Get请求。
    • 没有删除父类的方法。如父类的doGet方法会先获取一个版本号,根据版本号发送不同的错误,并会指定一个错误页面
  3. 500:服务器内部错误,代码出现了异常
  4. 空白页面:服务器没有给浏览器返回任何的body数值
  5. 无法访问此网站:Tomcat 服务器未正确运行/IP/端口号编写不对

五、关于自定义数据 ---- body或query String 里面的内容

5.1 地位

实际开发中,利用header的部分较少,更多的时候,我们是希望获取到 query string 或 body 的内容,因为这些内容是自定义的,可以供我们完成服务。

5.2 给服务器传递自定义数据

  1. 方式:HttpServletResponse有获取参数的方法,获取的参数就是我们传过去的这些数据。我们共有四种方式传递数据:a=x传递、Postman传递form表单、Postman传递JSON数据、ajax传递。
  2. a=x传递
    • 这个是把数据传到query string里
    • query string本身是键值对结构的数据,Tomcat 在收到这个请求后,就会把这个query string解析成Map这样的键值对,使用getparameter就可以根据key获取到value。
  3. Postman传递form表单:使用form表单的形式提交,此时的body也是键值对格式
    • 这个是把数据传到 body 里
    • 如果是通过post form表单的形式提交的请求,body此时就也是和query string一样的键值对格式
    • 如何发送数据
      在这里插入图片描述
  4. Postman传递JSON数据
    • 注意:JSON格式下,key一定是字符串格式,原则上key是不需要加“”,但是有些库/程序检查比较严格,是需要给key加上“”的.
    • Postman 构造的JSON数据需要给key加上“”,Ajax 构造的则不需要加上“”
      在这里插入图片描述
    • 类里面的属性为什么不使用private:本身,Jackson 会通过反射的方式,把User类里包含的public属性都获取到,此时就可以根据反射这里得到的“属性名字”,去JSON解析出来的键值对中进行匹配,如果匹配到了,就把value设置到刚才得到的属性中。而Jackson并不会直接针对private属性进行扫描,所以如果要使用private,需要提供getter和setter方法
class User{public String username;public String password;
}
@WebServlet("/JSON")
public class JSONServlet extends HttpServlet {@Overrideprotected void doGet(HttpServletRequest req, HttpServletResponse resp) throws ServletException, IOException {ObjectMapper objectMapper = new ObjectMapper();User user = objectMapper.readValue(req.getInputStream(), User.class);resp.getWriter().write(user.username);resp.getWriter().write("<br>");resp.getWriter().write(user.password);}
}
  1. ajax传递
    • 既可以传到 query string,也可以传递body中
    • 如果是传递到 query string 中,依旧可以使用 getParameter 来获取
    • 如果是使用 ajax Post 提交【不是 from表单的格式】,就需要使用 getInputStream 来获取

5.3 获取自定义数据

  1. 获取query string 里的值
    • 约定数据:约定前端传过来一个怎样的数据
      • 形如 【password = 】这样的请求(只说明 key,不说明 value),password的值将是“”,即一个空字符串。
      • 如果不传password(忽略key),password的值是null
    • 选择调用的方法:调用getParameterNames()、getParameter(String name)、getParameterValues(String name)方法
  2. 获取body(只考虑form表单的格式)
    • 约定数据:Post方法下,提交的数据才会在body里
    • 和query string一样,使用 getParameter(String name)方法获取数据
  3. 获取body(考虑为JSON格式)
    • 为什么要引入Jackson库
      Servlet 自身不能对JSON的数据进行解析,所以我们需要引入第三方库来解析body数据,把这里面的键值对还原成如Map一样【key-value】的形式。而能够实现解析操作的库很多,我们这里使用Jackson来解析。
    • 引入 Jackson 库:通过Maven引入,选择【Jackson Databind】。版本选哪个都行,但最好不要选择新版本,因为新版本往往没有经过时间充分的验证,可能会有问题
    • 如何使用Jackson库:一个类,两个方法
      • ObjectMapper类:对象映射,可以实现JSON数据和对象间的转换
      • JSON数据转类对象:readValue(InputStream src,JavaType valueType),该方法会先把JSON字符串解析成键值对,再放到Map中,然后根据参数填入的类对象,通过反射API知道这个类的属性名和类型,一次把这里的每个属性都取出来后,通过属性名字查询上述的Map,最后把得到的值赋值给这个类的属性。
      • 类对象转JSON数据:writeValueAsString(Object value)
    • 关于类和JSON格式的转换问题:因为是使用匹配来转换,所以构建类要求属性名和键值对相同

http://www.ppmy.cn/server/27986.html

相关文章

jupyter notebook导出pdf文件显示不了中文

找到文件index.tex.j2&#xff0c;我的在 C:\Users\Administrator\miniconda3\envs\opencv2\share\jupyter\nbconvert\templates\latex 我安装miniconda3并配置opencv2所需要的环境, 配置前 最后&#xff1a;用文本编辑器打开&#xff0c;修改图中article为ctexart&#xf…

原生IP和住宅IP有什么区别?

原生IP和住宅IP在多个方面存在显著的区别。 从定义和来源来看&#xff0c;原生IP是指未经NAT&#xff08;网络地址转换&#xff09;处理的真实、公网可路由的IP地址&#xff0c;它直接从互联网服务提供商&#xff08;ISP&#xff09;获得&#xff0c;而不是通过代理服务器或VP…

查看服务器网卡是千兆网卡还是万兆网卡方法

&#x1f341;博主简介&#xff1a; &#x1f3c5;云计算领域优质创作者 &#x1f3c5;2022年CSDN新星计划python赛道第一名 &#x1f3c5;2022年CSDN原力计划优质作者 &#x1f3c5;阿里云ACE认证高级工程师 &#x1f3c5;阿里云开发者社区专…

使用zookeeper作为分布式节点的配置中心

安装 wget https://archive.apache.org/dist/zookeeper/zookeeper-3.4.9/zookeeper-3.4.9.tar.gz tar xfzv zookeeper-3.4.9.tar.gz && mv zookeeper-3.4.9/conf/zoo_sample.cfg zookeeper-3.4.9/conf/zoo.cfg vim zoo_sample.cfg zookeeper-3.4.9/conf/zoo.cfg 把dat…

线上线下收银一体化,新零售POS系统引领连锁门店数字化转型-亿发

在市场竞争日益激烈的背景下&#xff0c;没有哪个商家能够永远屹立不倒。随着互联网技术的快速发展&#xff0c;传统的线下门店面临着来自电商和新零售的新型挑战。实体零售和传统电商都需要进行变革&#xff0c;都需要实现线上线下的融合。 传统零售在客户消费之后就与商家失…

react完整项目搭建的思路

react完整项目搭建的思路 react完整项目搭建的思路1.使用creacte-react-app初始化项目2.安装所需插件:路由、网络、样式、组件库3.reactjs目录结构组织4. 配置路径别名4.配置路由5.网络配置,对axios进行封装》获取当前环境变量 6.配置代理解决跨域7.配置使用iconfont8.状态管理…

缓存分享(1)——Guava Cache原理及最佳实践

Guava Cache原理及最佳实践 1. Guava Cache是什么1.1 简介1.2 核心功能1.3 适用场景 2. Guava Cache的使用2.1 创建LoadingCache缓存2.2 创建CallableCache缓存 缓存的种类有很多&#xff0c;需要根据不同的应用场景来选择不同的cache&#xff0c;比如分布式缓存如redis、memca…

ES与关系数据库的同步练习(hotel_admin)

目录 1 es与数据库同步的方法2 实践2.1 任务介绍2.2 MQ方面操作2.2.1 声明交换机队列并且绑定2.2.2 hotel_admin端web层设置mq发送消息2.3 hotel_demo端监听接受消息并执行es操作 1 es与数据库同步的方法 方式一&#xff1a;同步调用 优点&#xff1a;实现简单&#xff0c;粗…