Tomcat工作原理及简单模拟实现

Tomcat应该都不陌生,我们经常会把写好的代码打包放在Tomcat里并启动,然后在浏览器里就能愉快的调用我们写的代码来实现相应的功能了,那么Tomcat是如何工作的?

一、Tomcat工作原理

我们启动Tomcat时双击的startup.bat文件的主要作用是找到catalina.bat,并且把参数传递给它,而catalina.bat中有这样一段话:

Bootstrap.class是整个Tomcat 的入口,我们在Tomcat源码里找到这个类,其中就有我们经常使用的main方法:

这个类有两个作用 :

  • 初始化一个守护进程变量、加载类和相应参数;

  • 解析命令,并执行。

源码不过多赘述,我们在这里只需要把握整体架构,有兴趣的同学可以自己研究下源码。Tomcat的server.xml配置文件中可以对应构架图中位置,多层的表示可以配置多个:

即一个由 Server->Service->Engine->Host->Context 组成的结构,从里层向外层分别是:

Server:服务器Tomcat的顶级元素,它包含了所有东西。

Service:一组 Engine(引擎) 的集合,包括线程池 Executor 和连接器 Connector 的定义。

Engine(引擎):一个 Engine代表一个完整的 Servlet引擎,它接收来自Connector的请求,并决定传给哪个Host来处理。

Container(容器):Host、Context、Engine和Wraper都继承自Container接口,它们都是容器。

Connector(连接器):将Service和Container连接起来,注册到一个Service,把来自客户端的请求转发到Container。

Host:即虚拟主机,所谓的”一个虚拟主机”可简单理解为”一个网站”。

Context(上下文 ): 即 Web 应用程序,一个 Context 即对于一个 Web 应用程序。

Context容器直接管理Servlet的运行,Servlet会被其给包装成一个StandardWrapper类去运行。Wrapper负责管理一个Servlet的装载、初始化、执行以及资源回收,它是最底层容器。

比如现在有以下网址,根据“/”切割的链接就会定位到具体的处理逻辑上,且每个容器都有过滤功能。

二、Tomcat实现思路

下面只是简单实现效果,当浏览器访问对应地址时:

实现以上效果整体思路如下:

1.ServerSocket占用8080端口,用while(true)循环等待用户发请求。

2.拿到浏览器的请求,解析并返回URL地址,用I/O输入流读取本地磁盘上相应文件。

3.读取文件,不存在构建响应报文头、HTML正文内容,存在则写到浏览器端。

三、实现Tomcat

工程文件结构和pom.xml文件:

1.HttpServer核心处理类,用于接受用户请求,传递HTTP请求头信息,关闭容器:

public class HttpServer {// 用于判断是否需要关闭容器private boolean shutdown = false;public void acceptWait() {ServerSocket serverSocket = null;try {//端口号,最大链接数,ip地址serverSocket = new ServerSocket(8080, 1, InetAddress.getByName("127.0.0.1"));}catch (IOException e) {e.printStackTrace();System.exit(1); }// 等待用户发请求while (!shutdown) {try {Socket socket = serverSocket.accept();InputStream is = socket.getInputStream();OutputStream  os = socket.getOutputStream();// 接受请求参数Request request = new Request(is);request.parse();// 创建用于返回浏览器的对象Response response = new Response(os);response.setRequest(request);response.sendStaticResource();//关闭一次请求的socket,因为http请求就是采用短连接的方式socket.close();//如果请求地址是/shutdown  则关闭容器if(null != request){shutdown = request.getUrL().equals("/shutdown");}}catch (Exception e) {e.printStackTrace();continue;}}}public static void main(String[] args) {HttpServer server = new HttpServer();server.acceptWait();}
}

2.创建Request类,获取HTTP的请求头所有信息并截取URL地址返回:

public class Request {private InputStream is;private String url;public Request(InputStream input) {this.is = input;}public void parse() {//从socket中读取一个2048长度字符StringBuffer request = new StringBuffer(Response.BUFFER_SIZE);int i;byte[] buffer = new byte[Response.BUFFER_SIZE];try {i = is.read(buffer);}catch (IOException e) {e.printStackTrace();i = -1;}for (int j=0; j<i; j++) {request.append((char) buffer[j]);}//打印读取的socket中的内容System.out.print(request.toString());url = parseUrL(request.toString());}private String parseUrL(String requestString) {int index1, index2;index1 = requestString.indexOf(' ');//看socket获取请求头是否有值if (index1 != -1) {index2 = requestString.indexOf(' ', index1 + 1);if (index2 > index1)return requestString.substring(index1 + 1, index2);}return null;}public String getUrL() {return url;}
}

3.创建Response类,响应请求读取文件并写回到浏览器

public class Response {public static final int BUFFER_SIZE = 2048;//浏览器访问D盘的文件private static final String WEB_ROOT ="D:";private Request request;private OutputStream output;public Response(OutputStream output) {this.output = output;}public void setRequest(Request request) {this.request = request;}public void sendStaticResource() throws IOException {byte[] bytes = new byte[BUFFER_SIZE];FileInputStream fis = null;try {//拼接本地目录和浏览器端口号后面的目录File file = new File(WEB_ROOT, request.getUrL());//如果文件存在,且不是个目录if (file.exists() && !file.isDirectory()) {fis = new FileInputStream(file);int ch = fis.read(bytes, 0, BUFFER_SIZE);while (ch!=-1) {output.write(bytes, 0, ch);ch = fis.read(bytes, 0, BUFFER_SIZE);}}else {//文件不存在,返回给浏览器响应提示,这里可以拼接HTML任何元素String retMessage = "<h1>"+file.getName()+" file or directory not exists</h1>";String returnMessage ="HTTP/1.1 404 File Not Found\r\n" +"Content-Type: text/html\r\n" +"Content-Length: "+retMessage.length()+"\r\n" +"\r\n" +retMessage;output.write(returnMessage.getBytes());}}catch (Exception e) {System.out.println(e.toString() );}finally {if (fis!=null)fis.close();}}
}

四、扩展点

1.在WEB_INF文件夹下读取web.xml解析,通过请求名找到对应的类名,通过类名创建对象,用反射来初始化配置信息,如welcome页面,Servlet、servlet-mapping,filter,listener,启动加载级别等。

2.抽象Servlet类来转码处理请求和响应的业务。发过来的请求会有很多,也就意味着我们应该会有很多的Servlet,例如:RegisterServlet、LoginServlet等等还有很多其他的访问。可以用到类似于工厂模式的方法处理,随时产生很多的Servlet,来满足不同的功能性的请求。

3.使用多线程。本文的代码是死循环,且只能有一个链接,而现实中的情况是往往会有很多很多的客户端发请求,可以把每个浏览器的通信封装到一个线程当中。

近期热门文章

  • Java 最常见的 200+ 面试题

转发朋友圈,是对我最大的支持。

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

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

相关文章

苹果怎么调字体大小?详细教程在这,赶快get起来!

在我们使用苹果手机时&#xff0c;可能会遇到需要调整字体大小的情况。合适的字体大小能够提高阅读体验和舒适度&#xff0c;使得手机使用更加便捷。然而&#xff0c;还有很多用户并不知道苹果怎么调字体大小。别着急&#xff01;本文将为您详细介绍操作方法&#xff0c;帮助您…

PyQt5树形结构控件QTreeWidget操作

QTreeWidget 类根据预设的模型提供树形显示控件。 QTreeWidget 使用类似于 QListView 类的方式提供一种典型的基于 item 的树形交互方法类&#xff0c;该类基于QT的“模型/视图”结构&#xff0c;提供了默认的模型来支撑 item 的显示&#xff0c;这些 item 类为 QTreeWidgetIt…

Python3之logging输出写入日志

Python3之logging模块浅析 目录 Python3之logging模块浅析 简单用法日志与控制台同时输出 一个同时输出到屏幕、文件的完成例子日志文件截取日志重复打印问题解决 问题分析解决方案 1.使用不同的日志对象2.及时清理&#xff08;logger.handlers.clear&#xff09;3.使用前先判…

HashMap线程安全问题

HashMap是线程不安全的&#xff0c;在多线程环境下对某个对象中HashMap类型的实例变量进行操作时&#xff0c;可能会产生各种不符合预期的问题。本文详细说明一下HashMap存在的几个线程安全问题。注&#xff1a;以下基于JDK1.8HashMap原理请走传送门通过简单例子来探索HashMap原…

四、PyCharm PyQt5创建主窗口详细教程

1.打开PyCharm,新建工程MyMainTest 2.按照Tools-External Tools-QtDesigner,打开QT设计界面,保存窗体文件。

java8 lambda表达式实现自定义用户组件,Don't Repeat Yourself

2019独角兽企业重金招聘Python工程师标准>>> 一、用户组件的功能 使用java8 lambda表达式实现实现世界的一个例子&#xff1a;用户组件。此用户组件有以下几个操作&#xff1a;获取用户列表&#xff0c;获取单个用户&#xff0c;增加用户&#xff0c;删除用户&am…

如何在CDH5上部署Dolphin Scheduler 1.3.1

本文记录了在CDH5.16.2集群上集成Dolphin Scheduler 1.3.1的详细流程&#xff0c;特别注意一下MySQL数据库的连接串&#xff01; 1 文档编写目的 详细记录CDH5上Dolphin Scheduler 1.3.1的部署流程分布式部署Dolphin Scheduler 2 部署环境和依赖组件 为了适配CDH5上的Hive…

Kafka面试题全套整理 | 划重点要考!

做积极的人&#xff0c;而不是积极废人&#xff01;有很多人问过我要过Kafka相关的面试题&#xff0c;我一直懒得整理&#xff0c;这几天花了点时间&#xff0c;结合之前面试被问过的、别人咨询过的、我会问别人的进行了相关的整理&#xff0c;也就几十题&#xff0c;大家花个几…

关于Java基础你不得不会的34个问题

目录1. 面向对象和面向过程的区别2. Java 语言有哪些特点3. 关于 JVM JDK 和 JRE 最详细通俗的解答4. Oracle JDK 和 OpenJDK 的对比5. Java和C的区别6. 什么是 Java 程序的主类 应用程序和小程序的主类有何不同7. Java 应用程序与小程序之间有那些差别8. 字符型常量和字符串常…

Android--快速接入微信支付

前言 最近实习的时候要求我做支付模块&#xff0c;主要是介入支付宝支付和微信支付的。支付宝支付接入相对来说比较好做&#xff0c;官网文档也比较容易懂。但是做微信支付的时候&#xff0c;官网文档就有点懵逼了&#xff0c;不过慢慢读还是能够开通的。与是抽时间记录一下微…

PyQt5 QTreeWidget更改item项前的展开折叠三角图标

把下面代码放到对话框初始化函数中即可 # 设置树控件样式self.treeWidget.setStyleSheet("QTreeView::branch:open:has-children:!has-siblings,""QTreeView::branch:open:has-children:has-siblings {image: url(:/pic/images/minus.png);}""QTreeVi…

聊一聊开发常用小工具

作者&#xff1a;肥朝 来自&#xff1a;肥朝&#xff08;ID&#xff1a;feichao_java&#xff09;之前不少朋友问到平时怎么学习的.今天就简单介绍一下平时用到的一些小工具.首先我平时是用IDEA开发的,所以下面的介绍都是和IDEA相关的.本文主要介绍我平时在哪些场景使用,使用了…

YOLOv5的详细使用教程,以及使用yolov5训练自己的数据集

YOLOv5的详细使用教程&#xff0c;以及使用yolov5训练自己的数据集 文章目录&#xff1a; 1 安装环境依赖 1.1 克隆项目1.2 安装必要的环境依赖2 下载预训练模型和标注的数据集 2.1 下载预训练模型 2.1.1 执行脚本下载预训练模型2.1.2 直接下载预训练模型&#xff0c;然后保存…

六、PyQt5 QTreeWidget操作,获取树控件所有2级目录内容

新建pyqt5工程目录,添加树控件QTreeWidget,添加4个按钮,分别是添加 、删除、修改、打印树内容按钮,并添加按钮单击消息响应函数。 适用于QTreeWidget树控件只有2级目录内容情况 将树控件一级目录作为键,二级目录作为其对应键值存储到字典中。 # 树控件的字典,例如:dict…

多线程小抄集

Java中线程的状态 NEW、RUNNABLE&#xff08;RUNNING or READY&#xff09;、BLOCKED、WAITING、TIME_WAITING、TERMINATEDJava将操作系统中的运行和就绪两个状态合并称为运行状态。阻塞状态是线程阻塞在进入synchronized关键字修饰的方法或代码块&#xff08;获取锁&#xff0…

使用rancher-2.5.5部署单节点kubernetes集群

rancher2.5.x发布&#xff0c;除了新的feature外&#xff0c;更加简洁&#xff0c;体验/研究/熟悉一下&#xff0c;方便后续的落地使用。感觉&#xff1a;太爽了 相关版本信息&#xff1a; rancher&#xff1a;v2.5.5 helm3&#xff1a;v3.5.1 kubernetes&#xff1a;1.19.…

1-2.Win10系统利用Pycharm社区版安装Django搭建一个简单Python Web项目的步骤之二

七、在项目下新建 templates 路径 在工程上&#xff0c;右键&#xff0c;添加templates目录 注意*: 此目录下即用来存放我们的html文件&#xff1b; 此目录一般是与app的主目录是平级的。当然也可以建立在app的目录下&#xff0c;主要取决于你的模板&#xff0c;一般是全局…

k8s-自动横向伸缩pod 根据CPU使用率,QPS访问数监控指标

k8s-自动横向伸缩pod 与节点 简述 我们可以通过调高ReplicationController、 ReplicaSet、 Deployment等可伸缩资源的rep让cas字段&#xff0c; 来手动实现pod中应用的横向扩容。 我们也可以通过增加pod容器的资源请求和限制来纵向扩容pod (尽管目前该操作只能在pod创建时&am…

记一次蚂蚁金服的面试经历

2015在实习的时候&#xff0c;当时一个一起实习的朋友在2019年3月份的时候突然在微信上找我&#xff0c;问我要不要面试下蚂蚁金服。问了下相关信息才知道他在2018年11月的时候进到蚂蚁金服&#xff0c;现在招人就想到了我&#xff0c;问我要不要试一下。刚开始还是有所顾虑的&…

记一次阿里巴巴一面的经历

上周在拉勾上收到一个蚂蚁金服的大哥要我的简历&#xff0c;当时很惊讶&#xff0c;居然有蚂蚁金服的找到我&#xff0c;然后想都没想就给了。受宠若惊呀&#xff0c;我知道自己的水平跟阿里的差距有多远&#xff0c;以前一直没用勇气去投&#xff0c;连试试都不敢。这次居然主…