【Servlet篇2】Servlet的工作过程,Servlet的api——HttpServletRequest

news2024/11/15 4:52:26

 

一、Servlet的工作过程

二、Tomcat的初始化

步骤1:寻找到当前目录下面所有需要加载的Servlet(也就是类)

步骤2:根据类加载的结果创建实例(通过反射),并且放入集合当中

步骤3:实例创建好之后,调用Servlet的init()方法初始化一些属性

步骤4:创建TCP的socket,并且监听8080端口,等待客户端建立连接

步骤5:关闭服务器的时候,需要调用Servlet的destory()方法

 三、Tomcat处理请求

 步骤1:根据socket构造请求的对象(req)以及响应的对象(resp)

步骤2:判断请求的资源是否是一个静态的页面

步骤3:根据请求的URL找到处理请求的Servlet

步骤4:根据Servlet对象,来调用service方法

最后:简单概括一下Servlet的生命周期:

doPost方法乱码问题:

四、Servlet的核心api

HttpServletRequest类

 String getRequestURI()和String getRequestURL()

String getQueryString() 

     Enumeration getParameterNames()

      String getParameter(String name)

      String getHeader("name")

      InputStream getInputStream() 

      String getContextPath()

一、Servlet的工作过程

       需要了解的是,Sevlet是工作在应用层的,是一个应用程序。它是一个运行在用户态的的java进程

       第一步:客户端主机生产了http报文,然后经过一次封装,到达客户端的物理层。然后以光电信号的方式在网络当中传输。

       第二步:服务端主机在收到了对应的光电信号之后,进行了一次数据从物理层服务端的应用层的一次分用。

       第三步:服务端计算客户端的响应,然后在把数据通过response对象回写给客户端。


二、Tomcat的初始化

步骤1:寻找到当前目录下面所有需要加载的Servlet(也就是类)

       在这一篇文章当中,我们提到了如何把一个项目打包成一个war包。war包当中的文件都是.class为后缀名的。

        然后把这一个war包放入到Tomcat的webapps目录下面。双击startup.bat就可以运行Tomcat了。

【Servlet篇2】创建一个web项目_革凡成圣211的博客-CSDN博客Tomcat的web项目https://blog.csdn.net/weixin_56738054/article/details/129265315?spm=1001.2014.3001.5501         然后tomcat就会从webapps里找到那些.class对应的Servlet类,进行加载。

        可以看到,经过编译的Servlet,都变成了.class结尾的了。 伪代码实现:

 public void start(){
        //第一步:通过Tomcat加载所有的.class结尾的文件
        Class<Servlet>[] allServletClasses=Tomcat.load();
    }

步骤2:根据类加载的结果创建实例(通过反射),并且放入集合当中

         遍历上面的allServletClasses数组。并且在遍历的过程当中,每遍历一个,都通过反射创建一个Servlet的实例对象。创建完成之后,再次把这些实例添加到Servlet实例的集合当中。

          伪代码实现:


    /**
     *这是一个存放Servlet实例的集合
     */
    private final List<Servlet> instanceServlet=new ArrayList<>();

    public void start() throws InstantiationException, IllegalAccessException {
        //第一步:通过Tomcat加载webapps下面所有的.class结尾的文件
        Class<Servlet>[] allServletClasses=Tomcat.load();
        //第二步:根据类加载的结果创建Servlet的实例
        for(Class<Servlet> servletClass:allServletClasses){
            //根据class加载的结果,创建servlet的实例对象
            Servlet servlet= servletClass.newInstance();
            //把Servlet的实例对象存放到集合当中
            instanceServlet.add(servlet);
        }
    }

步骤3:实例创建好之后,调用Servlet的init()方法初始化一些属性

         伪代码:

for(Servlet servlet:instanceServlet){
      //init方法默认情况下面什么都不做
      //但是可以初始化一些属性
      servlet.init();
 }

       前面的文章当中,我们提到了,如果一个普通的类想成为Servlet,那就一定需要首先继承于HttpServlet

       当然,也可以由当前的类初始化一些属性,那就需要重写init()方法了。

       每一个Servlet在启动的时候,都仅仅只会调用一次init()方法。


步骤4:创建TCP的socket,并且监听8080端口,等待客户端建立连接

 在服务端,使用线程池的的方式来处理每一个请求。

 在下面的伪代码当中,首先使用serverSocket监听8080端口。

然后服务端重复调用accept()方法等待客户端建立连接

并且使用往线程池当中提交任务的方式来处理请求

伪代码实现:

//这个socket是服务端TCP的socket,创建的时候默认占用8080端口
        ServerSocket serverSocket=new ServerSocket(8080);
        //使用线程池,添加任务
        ExecutorService service= Executors.newFixedThreadPool(10);
        while (true){
            //每当有一个客户端与当前的服务端建立连接的时候,就会监听8080端口
            //accept()一次
            Socket socket=serverSocket.accept();
            //向线程池提交任务
            service.execute(new Runnable() {
                @Override
                public void run() {
                   doHttpRequest(socket);
                }
            });
        }

其中,doHttpRequest方法内部,就是解析客户端发送请求的方法。也就是执行doGet被调用的过程。


步骤5:关闭服务器的时候,需要调用Servlet的destory()方法

 伪代码实现:

 //关闭服务器,销毁servlet
 for(Servlet servlet:instanceServlet){
        servlet.destroy();
  }

总览一下:Tomcat初始化的过程:


 三、Tomcat处理请求

 刚刚提到了,在doHttpServletRequest方法当中,是对于socket进行处理的过程。

 下面,继续通过伪代码的方式,来解析一下具体是怎样处理请求的。

 步骤1:根据socket构造请求的对象(req)以及响应的对象(resp)

 对于request:首先读取socket当中的数据,然后再按照HTTP协议的格式来返回。

 对于response:仅仅只是构造了一个空的对象,还没有填充响应的信息

伪代码: 

private void doHttpRequest(Socket socket) {
        //对于request,首先读取socket当中的数据,然后再按照HTTP协议的格式来返回
        HttpServletRequest request=HttpServletRequest.parse(socket);
        //构造一个没有填充数据是response
        HttpServletResponse response=HttpServletResponse.build(socket);
    }

步骤2:判断请求的资源是否是一个静态的页面

伪代码:

       如果请求的资源是一个静态的页面,例如:https:....../.html这样的页面。那么就读取文件的内容,并且把文件的内容构造的resp对象当中。

【网络原理9】HTTP响应篇_革凡成圣211的博客-CSDN博客HTTP响应,HTTP状态码https://blog.csdn.net/weixin_56738054/article/details/129208502?spm=1001.2014.3001.5502         在HTTP响应篇这一篇文章当中提到了:HTTP响应头当中有一个属性代表的是返回的资源的类型(content type属性)。如果是一个静态的html页面,那么这个类型的值就是test/html

//访问的资源是一个静态的页面
        if(file.exists()){
            //通过response返回这一个页面
            return;
        }

步骤3:根据请求的URL找到处理请求的Servlet

伪代码:

//通过请求的URL路径,返回Tomcat加载的对应的那一个Servlet
Servlet ins=findInstance(request.getRequestURL());

 例如一个请求是:localhost:8080/hello106/hello

 那么这个hello代表的就是一个Servlet


步骤4:根据Servlet对象,来调用service方法

伪代码:

    try {
          //调用service方法
         ins.service(request,response);
        }catch (Exception e){

     }

其实在service方法内部,就是根据request的请求来判断当前的请求的方法是属于get还是post

如果是get请求,那么就交给doGet方法来进行处理。

如果是post请求,那么就交给doPost方法来及逆行处理。


下面,通过一张图,来总览一下Tomcat处理请求的各个步骤

最后:简单概括一下Servlet的生命周期:

当被tomcat初始化的时候:调用init方法完成一些属性的初始化操作;

当执行具体逻辑的时候:调用service方法,然后在内部判断是doGet还是doPost方法;

当执行完毕的时候:调用destroy方法销毁。


doPost方法乱码问题:

在一个类当中,首先让一个类继承于HttpServlet这一个类,然后重写doPost方法。

并且通过resp.getWriter().write(字符串内容);来在页面上面输出一段话:

/**
 * @author 革凡成圣211
 */
@WebServlet("/method")
public class Servlet2 extends HttpServlet {

    @Override
    protected void doPost(HttpServletRequest req, HttpServletResponse resp) throws ServletException, IOException {
        resp.getWriter().write("POST响应");
    }
}

然后在前端页面当中通过post来提交请求(注意不可以直接在url地址栏当中输入,否则默认是get)

 

 但是看到,此处的中文出现了乱码:

  首先,分析一下,在当前的场景当中,字符串生成的两个环节:

  环节1:生成(idea里面通过硬编码的方式把这个字符串写进去)

  环节2:展示(浏览器把这个字符串显示到控制台当中)

如果以上的两个操作不一致,那么就比较容易出现乱码。

对于环节2,浏览器展示的时候,一般都是默认使用windows简体的中文版,默认是gbk

对于环节1,idea编码的方式一般都是使用utf-8

为了统一idea浏览器的编码方式,可以改变浏览器的读取的编码规则,让浏览器按照utf-8来读取。

/**
 * @author 革凡成圣211
 */
@WebServlet("/hello")
public class HelloServlet extends HttpServlet {


    @Override
    protected void doPost(HttpServletRequest req, HttpServletResponse resp) throws ServletException, IOException {
        //设置响应头的响应格式,以及字符集编码方式
        resp.setContentType("text/html; charset=utf8");
        resp.getWriter().write("POST"+"响应");

    }
}

 也就是把HTTP响应头当中的两个键值对,一个是content-type,另外一个是charset

 不过,也有更简单的方式来发送post请求,那就是使用postman。

HttpServletRequest类

       每一个HttpServletRequest代表一个HTTP请求。也就是一个HTTP请求当中包含什么内容,那么一个HttpServletRequest就封装了什么内容。

       在这一篇文章当中,我们提到了:使用Fiddler抓包的时候,提到了一个HTTP请求是由三部分组成的:【网络原理7】认识HTTP_革凡成圣211的博客-CSDN博客HTTP抓包,Fiddler的使用https://blog.csdn.net/weixin_56738054/article/details/129148515?spm=1001.2014.3001.5502

   第一部分:请求行(URL)

   第二部分:请求头(多组键值对)

   第三部分:请求的body

        那么,通过一个HttpServletRequest同样也可以获取到一个HTTP请求当中的各类信息。


       下面,介绍几个HttpServletRequest的几个常用方法:

 String getRequestURI()和String getRequestURL()

       假如我想访问的资源是:http://localhost:8080/ServletLearning_war_exploded/hello

方法名称返回含义
String getRequestURI()"/ServletLearning_war_exploded/hello"除了域名+端口号往后的内容,也就是需要访问的具体是哪一个文件
String getRequestURL()"http://localhost:8080/ServletLearning_war_exploded/hello"一个完整的请求路径

       需要注意的是,无论是URI还是URL,请求路径当中如果包含了queryString(查询字符串)

       那么都无法在上述两个方法当中返回。

       


String getQueryString() 

       这个方法,仅仅在URL地址栏当中包含queryString的时候,才会生效。

       假如我想访问呢的URL地址为:

        http://localhost:8080/ServletLearning_war_exploded/hello?key=666

       可以看到,在上述的URL当中,携带了一个queryString,那就是key=666

        所以上述方法返回的就是:

        "key=666"这一个字符串


     Enumeration getParameterNames()

      也是在请求栏当中存在queryString的时候生效。得到的所有key,都是以枚举的方式来列举的

       


      String getParameter(String name)

       根据根据queryString的key来获取到value


      String getHeader("name")

       这一个方法是获取HTTP请求头当中多组键值对中,其中以"name"为key的value。

       例如: req.getHeader("Content-Type")

       那么,这个方法就是返回Content-Type的值是什么

       

       例如在上述的方法当中,返回的内容就是:application/x-www-form-urlencoded

       


      InputStream getInputStream() 

       可以通过这个输入流对象,读取到HTTP请求的body部分的内容。


      String getContextPath()

       这一个方法返回的是一个项目的名称

       例如一个URL请求为:http://localhost:8080/ServletLearning_war_exploded/hello

       那么返回的内容就是:/ServletLearning_war_exploded


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

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

相关文章

SpringBoot接口+Redis解决用户重复提交问题

前言 1. 为什么会出现用户重复提交 网络延迟的情况下用户多次点击submit按钮导致表单重复提交&#xff1b;用户提交表单后&#xff0c;点击【刷新】按钮导致表单重复提交&#xff08;点击浏览器的刷新按钮&#xff0c;就是把浏览器上次做的事情再做一次&#xff0c;因为这样也…

前后端分页查询好大的一个坑(已解决)

前言&#xff1a;如果你在做前后端的分页查询&#xff0c;找不到错误&#xff0c;请你来看看是否是和我一样的情况&#xff1f;情况&#xff1a;做了一个前后盾UI的项目&#xff0c;有一个页面是查询系统日志&#xff0c;要进行分页查询&#xff1b;第一页的&#xff1a;第5页的…

MYSQL 基础篇 | 02-MYSQL基础应用

文章目录1 MySQL概述2 SQL2.1 SQL通用语法2.2 SQL分类2.3 DDL2.3.1 数据库操作2.3.2 表操作2.4 DML2.4.1 添加数据2.4.2 修改数据2.4.3 删除数据2.5 DQL2.5.1 基础查询2.5.2 条件查询2.5.3 聚合查询2.5.4 分组查询2.5.5 排序查询2.5.6 分页查询2.5.7 综合练习2.6 DCL2.6.1 管理…

SQLMAP使用

SQLMAPSQLMAP是什么SQLMAP可以对URL做什么&#xff1f;SQLMAP支持的注入技术SQLMAP检测注入漏洞流程&#xff1a;SQLMAP的误报检测机制SQLMAP基本使用常见用法常见参数tamper参数操作系统权限参数文件读写参数SQLMAP是什么 SQLMAP是一个开源的自动化SQL注入工具&#xff0c;主…

SAP 生产订单/流程订单中日期的解释

SAP 生产订单/流程订单中日期的解释 基本开始日期&#xff1a;表示订单的开始日期 基本完成日期&#xff1a;表示订单的完成日期 我们在输入基本开始日期和基本完成日期时需要关注 调度 下面的“类型”&#xff0c;其中有向前、向后、当天日期等&#xff1a; 调度类型 为向前…

上海交大陈海波教授、夏虞斌教授领衔巨作上市:《操作系统:原理与实现》

❤️作者主页&#xff1a;小虚竹 ❤️作者简介&#xff1a;大家好,我是小虚竹。2022年度博客之星评选TOP 10&#x1f3c6;&#xff0c;Java领域优质创作者&#x1f3c6;&#xff0c;CSDN博客专家&#x1f3c6;&#xff0c;华为云享专家&#x1f3c6;&#xff0c;掘金年度人气作…

java 进阶—线程的常用方法

大家好&#xff0c;通过java进阶—多线程&#xff0c;我们知道的什么是进程&#xff0c;什么是线程&#xff0c;以及线程的三种创建方式的选择 今天&#xff0c;我们来看看线程的基础操作 start() 开启线程 public class Demo implements Runnable {Overridepublic void run…

第五回:样式色彩秀芳华

import matplotlib as mpl import matplotlib.pyplot as plt import numpy as np第五回详细介绍matplotlib中样式和颜色的使用&#xff0c;绘图样式和颜色是丰富可视化图表的重要手段&#xff0c;因此熟练掌握本章可以让可视化图表变得更美观&#xff0c;突出重点和凸显艺术性。…

施工机械设备群远程在线监控管理系统

一、项目背景 在加强基础设施建设等一系列政策的牵引下&#xff0c;我国工程机械设备市场连续保持强劲增长势头&#xff0c;伴随国内中高端机械设备公司业务的不断扩展&#xff0c;施工大型机械设备的应用率不断提高&#xff0c;铁路施工特别是架桥机作业过程被认为是一个极易出…

元宇宙到底是什么?元宇宙解读,这一篇就够了!

欢迎来到Hubbleverse &#x1f30d; 关注我们 关注宇宙新鲜事 &#x1f4cc; 预计阅读时长&#xff1a;12分钟 本文仅代表作者个人观点&#xff0c;不代表平台意见&#xff0c;不构成投资建议。 在元宇宙中&#xff0c;我们被承诺可以成为任何人&#xff0c;做任何事&#…

JavaSE20-集合1-list

文章目录一、集合概念二、list集合1、list集合特点2、ArrayList2.1 创建对象2.2 常用方法2.3 遍历2.3.1 使用索引遍历2.3.2 使用迭代器遍历2.3.3 使用foreach遍历2.3.4 集合转换为数组遍历3、LinkedList3.1 创建对象3.2 常用方法3.3 遍历4、ArrayList和LinkedList的区别一、集合…

支持DDR5,超频更简单,小雕够给力,技嘉B760M小雕WIFI主板上手

目前13代酷睿已经全员集结了&#xff0c;其中全新的i5 13490F应该依然会备受欢迎&#xff0c;当然了&#xff0c;刚上市不久的13代酷睿价格方面还不是很有吸引力&#xff0c;好在12代酷睿在新一代主板上面依然可用&#xff0c;所以预算有限的朋友&#xff0c;完全可用继续使用1…

原生微信小程序引入npm和安装Vant Weapp

目录一、引入npm安装Vant Weapp1、引入npm2、安装Vant Weapp3、修改 app.json4、修改 project.config.json二、构建npm一、引入npm安装Vant Weapp 环境&#xff1a;Windows10 开发工具&#xff1a;微信开发者工具 本地环境&#xff1a;已安装过node.js 1、引入npm cmd进入到你…

MyBatis源码分析(二、续)SqlSource创建流程,SQL如何解析?如何将#{id}变成?的

文章目录实例一、SqlSource处理入口二、SqlSource处理逻辑1、XMLScriptBuilder 构造方法2、解析动态sql3、DynamicSqlSource4、RawSqlSource解析sql&#xff08;1&#xff09;parse方法解析sql写在后面实例 此处我们分析的sql&#xff1a; <select id"selectBlog&quo…

|干货 | 五种常用类型之String字符串详解

一. 背景说明小白&#xff1a;哥&#xff0c;java中String是最常用类型&#xff0c;Redis中也是吗?哥&#xff1a;差不多&#xff0c;我给你稍微讲一下。二. 数据类型依据Redis官网&#xff0c;目前Redis数据类型共计九种。具体整理如下&#xff1a;常用的数据类型有&#xff…

第九章 实现isReactive和isReadonly

实现isReactive和isReadonly isReactive实现 先上测试用例&#xff08;其实这个测试用例也是reactive.spec.ts中追加的两个&#xff09;: import { isReactive, reactive } from "../reactive"describe(reactive,()>{it(happy path,()>{const original {fo…

taobao.item.sku.add( 添加SKU )

&#xffe5;开放平台免费API必须用户授权 新增一个sku到num_iid指定的商品中 传入的iid所对应的商品必须属于当前会话的用户 公共参数 请求地址: HTTP地址 http://gw.api.taobao.com/router/rest 公共请求参数: 公共响应参数: 请求参数 响应参数 点击获取key和secret 请求…

第十四节 包、权限修饰符、final、常量

包 1.同一个包下的类&#xff0c;相互可以直接访问。 2.不同包下的类要导包后才能访问。 AIT回车键导包。 权限修饰符 什么是权限修饰符? ●权限修饰符:是用来控制一个成员能够被访问的范围。 ●可以修饰成员变量&#xff0c;方法&#xff0c;构造器&#xff0c;内部类&…

并发就一定快吗?答:肯定不是啊

文章目录一、多线程概念1.1 程序的并发与并行1.1.1 程序的并行1.1.2 程序的并发1.2 进程与线程1.2.1 进程1.2.2 线程1.2.3 多线程并发就一定快吗&#xff1f;答案直接戳这里&#x1f449;&#xff1a;多线程并发就一定快吗&#xff1f; 一、多线程概念 在实际应用中&#xff…

车载测试之电子设备有哪些测试点?

现在车上大大小小的控制器大几十个&#xff0c;主机厂都要做哪些测试&#xff0c;满足哪些要求&#xff0c;才能使控制器达到量产要求呢&#xff1f; 整车开发流程 在聊测试之前&#xff0c;首先了解一下一款车的开发流程。在主机厂我们经常能听到“某某项目开G2阀”、“某某…