tomcat原理解析(一):一个简单的实现

一 概述

       前段时间去面试,被人问到了tomcat实现原理。由于平时没怎么关注容器的实现细节,这个问题基本没回答上来。所以最近花了很多时间一直在网上找资料和看tomcat的源码来研究里面处理一个HTTP请求的流程。网上讲tomcat的帖子比较多,大多都是直接切入主题看其源码,从我个人感受来说直接研究其源码实现比较难理解和非常枯燥,需要由简到难,慢慢深入。

二  一个简单tomcat服务器实现

        tomat是一个servlet容器,来处理http请求。在平时的使用中我们都会再浏览器中输入http地址来访问服务资源,比如格式http://host[":"port][abs_path]。从浏览器到服务端的一次请求都遵循http协议,在网络上其实走仍然是tcp协议,即我们常使用的socket来处理客户端和服务器的交互。根据输入的http地址可以知道服务器的IP地址和端口,根据这两个参数就可以定位到服务器的唯一地址。tomcat根据http地址端口后面的资源路径就可以知道反馈什么样的资源给浏览器。下面给出了一个非常简单的代码模拟了tomcat的简单实现

 

[html] view plaincopy
  1. package com;  
  2.   
  3. import java.io.*;  
  4. import java.net.ServerSocket;  
  5. import java.net.Socket;  
  6. import java.net.URLDecoder;  
  7. import java.util.StringTokenizer;  
  8.   
  9. public class TomcatServer {  
  10.   
  11.     private final static int PORT = 8080;  
  12.   
  13.     public static void main(String[] args) {  
  14.   
  15.         try {  
  16.             ServerSocket server = new ServerSocket(PORT);//根据端口号启动一个serverSocket  
  17.             ServletHandler servletHandler=new ServletHandler(server);  
  18.             servletHandler.start();  
  19.         } catch (Exception e) {  
  20.             e.printStackTrace();  
  21.         }  
  22.   
  23.     }  
  24.   
  25.   
  26.   
  27.     private static class ServletHandler extends Thread{  
  28.         ServerSocket server=null;  
  29.         public ServletHandler(ServerSocket server){  
  30.             this.server=server;  
  31.         }  
  32.   
  33.   
  34.         @Override  
  35.         public void run() {  
  36.             while (true) {  
  37.                 try {  
  38.                     Socket client = null;  
  39.                     client = server.accept();//ServerSocket阻塞等待客户端请求数据  
  40.                     if (client != null) {  
  41.                         try {  
  42.                             System.out.println("接收到一个客户端的请求");  
  43.   
  44.                             //根据客户端的Socket对象获取输入流对象。  
  45.                             //封装字节流到字符流  
  46.                             BufferedReader reader = new BufferedReader(new InputStreamReader(client.getInputStream()));  
  47.   
  48.                             // GET /test.jpg /HTTP1.1  
  49.                             //http请求由三部分组成,分别是:请求行、消息报头、请求正文。  
  50.                             //这里取的第一行数据就是请求行。http协议详解可以参考http://www.cnblogs.com/li0803/archive/2008/11/03/1324746.html说的很详细  
  51.                             String line = reader.readLine();  
  52.   
  53.                             System.out.println("line: " + line);  
  54.   
  55.                             //拆分http请求路径,取http需要请求的资源完整路径  
  56.                             String resource = line.substring(line.indexOf('/'),line.lastIndexOf('/') - 5);  
  57.   
  58.                             System.out.println("the resource you request is: "+ resource);  
  59.   
  60.                             resource = URLDecoder.decode(resource, "UTF-8");  
  61.   
  62.                             //获取到这次请求的方法类型,比如get或post请求  
  63.                             String method = new StringTokenizer(line).nextElement().toString();  
  64.   
  65.                             System.out.println("the request method you send is: "+ method);  
  66.   
  67.                             //继续循环读取浏览器客户端发出的一行一行的数据  
  68.                             while ((line = reader.readLine()) != null) {  
  69.                                 if (line.equals("")) {//当line等于空行的时候标志Header消息结束  
  70.                                     break;  
  71.                                 }  
  72.                                 System.out.println("the Http Header is : " + line);  
  73.                             }  
  74.   
  75.                             //如果是POST的请求,直接打印POST提交上来的数据  
  76.                             if ("post".equals(method.toLowerCase())) {  
  77.                                 System.out.println("the post request body is: "  
  78.                                         + reader.readLine());  
  79.                             }else if("get".equals(method.toLowerCase())){  
  80.                                 //判断是get类型的http请求处理  
  81.                                 //根据http请求的资源后缀名来确定返回数据  
  82.   
  83.                                 //比如下载一个图片文件,我这里直接给定一个图片路径来模拟下载的情况  
  84.                                 if (resource.endsWith(".jpg")) {  
  85.                                     transferFileHandle("d://123.jpg", client);  
  86.                                     closeSocket(client);  
  87.                                     continue;  
  88.   
  89.                                 } else {  
  90.   
  91.                              //直接返回一个网页数据  
  92.                              //其实就是将html的代码以字节流的形式写到IO中反馈给客户端浏览器。  
  93.                              //浏览器会根据http报文“Content-Type”来知道反馈给浏览器的数据是什么格式的,并进行什么样的处理  
  94.   
  95.                              PrintStream writer = new PrintStream(client.getOutputStream(), true);  
  96.                              writer.println("HTTP/1.0 200 OK");// 返回应答消息,并结束应答  
  97.                              writer.println("Content-Type:text/html;charset=utf-8");  
  98.                              writer.println();  
  99.                              //writer.println("Content-Length:" + html.getBytes().length);// 返回内容字节数  
  100.                              writer.println("<html><body>");  
  101.                              writer.println("<href='www.baidu.com'>百度</a>");  
  102.                              writer.println("<img src='https://ss0.bdstatic.com/5aV1bjqh_Q23odCf/static/superman/img/logo/bd_logo1_31bdc765.png'></img>");  
  103.                              writer.println("</html></body>");  
  104.   
  105.   
  106.                              //writer.println("HTTP/1.0 404 Not found");// 返回应答消息,并结束应答  
  107.                              writer.println();// 根据 HTTP 协议, 空行将结束头信息  
  108.                              writer.close();  
  109.                              closeSocket(client);//请求资源处理完毕,关闭socket链接  
  110.                              continue;  
  111.                                 }  
  112.                             }  
  113.   
  114.   
  115.   
  116.                         } catch (Exception e) {  
  117.                             System.out.println("HTTP服务器错误:"  
  118.                                     + e.getLocalizedMessage());  
  119.                         }  
  120.                     }  
  121.                 } catch (Exception e) {  
  122.                     e.printStackTrace();  
  123.                 }  
  124.             }  
  125.         }  
  126.   
  127.         private void closeSocket(Socket socket) {  
  128.             try {  
  129.                 socket.close();  
  130.             } catch (IOException ex) {  
  131.                 ex.printStackTrace();  
  132.             }  
  133.             System.out.println(socket + "离开了HTTP服务器");  
  134.         }  
  135.   
  136.         private void transferFileHandle(String path, Socket client) {  
  137.   
  138.             File fileToSend = new File(path);  
  139.   
  140.             if (fileToSend.exists() && !fileToSend.isDirectory()) {  
  141.                 try {  
  142.                     //根据Socket获取输出流对象,将访问的资源数据写入到输出流中  
  143.                     PrintStream writer = new PrintStream(client.getOutputStream());  
  144.                     writer.println("HTTP/1.0 200 OK");// 返回应答消息,并结束应答  
  145.                     writer.println("Content-Type:application/binary");  
  146.                     writer.println("Content-Length:" + fileToSend.length());// 返回内容字节数  
  147.                     writer.println();// 根据 HTTP 协议, 空行将结束头信息  
  148.   
  149.                     FileInputStream fis = new FileInputStream(fileToSend);  
  150.                     byte[] buf = new byte[fis.available()];  
  151.                     fis.read(buf);  
  152.                     writer.write(buf);  
  153.                     writer.close();  
  154.                     fis.close();  
  155.                 } catch (IOException e) {  
  156.                     e.printStackTrace();  
  157.                 }  
  158.             }  
  159.         }  
  160.   
  161.     }  
  162.   
  163. }  

 

三  实践

    1.在浏览器中输入http://localhost:8080/123.jpg 链接,可以看到浏览器里面就将123.jpg下载到本地了。

    2.在浏览器中输入一个服务器不能识别的请求后缀比如http://localhost:8080/123.jpg1,可以看到浏览器打开了一个网页。如下图:点击里面的百度链接可以跳转

    3.后台tomcat服务器打印的http请求报文

 

      接收到一个客户端的请求
line: GET /123.jpg1 HTTP/1.1
the resource you request is: /123.jpg1
the request method you send is: GET
the Http Header is : Host: localhost:8080
the Http Header is : Connection: keep-alive
the Http Header is : Pragma: no-cache
the Http Header is : Cache-Control: no-cache
the Http Header is : Accept: text/html,application/xhtml+xml,application/xml;q=0.9,image/webp,*/*;q=0.8
the Http Header is : Upgrade-Insecure-Requests: 1
the Http Header is : User-Agent: Mozilla/5.0 (Windows NT 6.1; WOW64) AppleWebKit/537.36 (KHTML, like Gecko) Chrome/47.0.2526.106 Safari/537.36
the Http Header is : Accept-Encoding: gzip, deflate, sdch
the Http Header is : Accept-Language: zh-CN,zh;q=0.8
Socket[addr=/0:0:0:0:0:0:0:1,port=57864,localport=8080]离开了HTTP服务器

四  总结

从整个代码和测试情况来看,一次http请求其实就是一次socket套接字的处理。浏览器发起scoket的请求,tomcat服务器接受请求,并根据请求的路径定位客户端需要访问的资源。  只是socket客户端和服务器数据在交互时,都遵守着http协议规范。当然真正的tomcat容器比这个demo实现要复杂的很多,这个简易的tomcat服务器能够帮我们更好的理解tomcat源码。

转载于:https://www.cnblogs.com/csguo/p/7499395.html

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

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

相关文章

手风琴案例jquery写法

今天我用jquary来写一下手风琴案例&#xff0c;这个案例在平时的项目中很经常会见到&#xff0c;要想实现效果用jquary来写其实很简单&#xff0c;其实一句话就是jquary的方法的调用。 首先我们先来分析一下手风琴案例实际实现的效果&#xff0c;就是点击当前的标题&#xff0c…

元组tuple

另一种有序列表叫元组&#xff1a;tuple。tuple和list非常类似&#xff0c;但是tuple一旦初始化就不能修改&#xff0c;比如同样是列出同学的名字&#xff1a; >>> classmates (Michael, Bob, Tracy)现在&#xff0c;classmates这个tuple不能变了&#xff0c;它也没有…

win7下的nginx小demo

一直大概知道nginx怎么玩,但是不看文档又蒙蔽.在这记录一下,以后好查看 下载tomcat,改index.jsp http://tomcat.apache.org/download-80.cgi tomcat9已经出来了,但是自己用了一次,闪退,换tomcat8,开启成功.(tomcat9这个原因有时间在琢磨) 修改tomcat的index.jsp 然后在index.js…

understand的安装

1.win7 64位下安装 1&#xff09;下载Understand.4.0.908.x64.rar。 2&#xff09;解压之&#xff0c;直接运行里面的Understand-4.0.908-Windows-64bit.exe。 3&#xff09;选择如下 之后&#xff0c;点击“Add Eval or SDL (RegCode)”&#xff0c;如下图&#xff1a; 4&…

条件、循环、函数定义 练习

a.五角星 import turtleturtle.color(yellow)turtle.begin_fill()for i in range(5): turtle.forward(100) turtle.right(144)turtle.end_fill() b.同心圆 import turtlefor i in range(5): turtle.up() turtle.goto(0,-20*(i1)) turtle.down() turtle.circle(20*(i1)) c.太阳花…

2015年上半年 软件设计师 上午试卷 综合知识-2

2015年上半年 软件设计师 上午试卷 综合知识-2 与算术表达式"&#xff08;a&#xff08;b-c&#xff09;&#xff09;*d" 对应的树是&#xff08;21&#xff09;。 答案&#xff1a; B 本题考查程序语言与数据结构基础知识。 对算术表达式"(a(b-c))*d"求…

Python web开发——自定义userprofile(用户描述)

1、新建一个APP 2、查看数据库中系统给我们提供的默认的users的字段含义 ID&#xff1a; 是主键&#xff0c;用户的ID passWord&#xff1a;密码 last_login : 最后一次登录的时间 is_superuser&#xff1a;是否是超级用户&#xff08;VIP&#xff09; username&#xff1a;用户…

Android之View绘制流程开胃菜---setContentView(...)详细分析

版权声明&#xff1a;本文出自汪磊的博客&#xff0c;转载请务必注明出处。 1 为什么要分析setContentView方法 作为安卓开发者相信大部分都有意或者无意看过如下图示&#xff1a;PhoneWindow,DecorView这些究竟都是些神马玩意&#xff1f;图示的层级关系是怎么来的&#xff1f…

排序算法之堆排序

一、什么是堆 如果一个完全二叉树的每个节点&#xff0c;都不大于它的子节点&#xff0c;就可以称之为堆。所谓完全二叉树&#xff0c;就是除了叶子节点以外&#xff0c;所有的其他节点&#xff0c;都有完整的左字树和右子树&#xff0c;除了最后一层的非叶子节点以外。 二、堆…

Codeforces Round #434 (Div. 2)【A、B、C、D】

Codeforces Round #434 (Div. 2) codeforces 858A. k-rounding【水】 题意&#xff1a;已知n和k&#xff0c;求n的最小倍数x&#xff0c;要求x后缀至少有k个0。 题解&#xff1a;答案就是10^k和n的最小公倍数。 1 #include<cstdio>2 #include<cstring>3 #include&l…

PLC与触摸屏练习

电机正反转 触摸屏 要用触摸屏进行仿真是不要忘了先启动梯形图测试 效果 1-2例 行程开关控制的自动循环 梯形图 触摸屏 下面只是用了四个开关&#xff0c;可以根据自己想要实现的按照梯形图添加 题目 梯形图 电动机星三角减压启动控制 题目及要求 触摸屏截图 运算 把显示的数值…

Django里面是文件静态化的方法

看Django官网的时候&#xff0c;由于自己的英语基础较差&#xff0c;而实现的谷歌翻译比较烂&#xff0c;只能看懂个大概。在文件静态化的时候&#xff0c;讲的比较繁琐一点&#xff0c;没怎么看懂&#xff0c;遂询问了一下其他人&#xff0c;明白了许多&#xff0c;但是细节需…

RabbitMQ 声明Queue时的参数们的Power

RabbitMQ 声明Queue时的参数们的Power 参数们的Power 在声明队列的时候会有很多的参数 public static QueueDeclareOk QueueDeclare(this IModel model, string queue "", bool durable false, bool exclusive true, bool autoDelete true, IDictionary<strin…

解决Firefox已阻止运行早期版本Adobe Flash

解决Firefox已阻止运行早期版本Adobe Flash 类别 [随笔分类]web 解决Firefox已阻止运行早期版本Adobe Flash 最近火狐浏览器不知抽什么风&#xff0c;每次打开总提示"Firefox已阻止(null)运行早期版本的Adobe Flash"。要命的是它提示的解决办法根本不管用&#xf…

使用MyBatista----上传图像

使用MyBatis上传图像&#xff0c;使用的是Oracle的数据库表&#xff0c;有一个TEACHER表&#xff0c;有7列&#xff0c;有1列是存储图片的&#xff0c;类型用BLOB&#xff0c;最大容量是4G&#xff0c;以二进制的形式写入数据库表。 建立这个表的对应实体类Teacher&#xff0c;…

BZOJ 2768 [JLOI2010]冠军调查

还说还剩十分钟A一道水题&#xff0c;然后发现和善意的投票一模一样粘个代码过去直接A。。。 装作自己又写了一道题。 题面 //Twenty #include<cstdio> #include<cstdlib> #include<iostream> #include<algorithm> #include<cmath> #include<…

【OpenCV函数】轮廓提取;轮廓绘制;轮廓面积;外接矩形

FindContours 在二值图像中寻找轮廓 int cvFindContours( CvArr* image, CvMemStorage* storage, CvSeq** first_contour, int header_sizesizeof(CvContour), int modeCV_RETR_LIST, int methodCV_CHAIN_APPROX_SIMPLE, CvPoint offsetcvPoint(0,0) ); image 输入的 8-比…

Windows下编译TensorFlow1.3 C++ library及创建一个简单的TensorFlow C++程序

由于最近比较忙&#xff0c;一直到假期才有空&#xff0c;因此将自己学到的知识进行分享。如果有不对的地方&#xff0c;请指出&#xff0c;谢谢&#xff01;目前深度学习越来越火&#xff0c;学习、使用tensorflow的相关工作者也越来越多。最近在研究tensorflow线下采用 pytho…

粉红小猪中有一个叫“快乐小鸡”的游戏

最近在学习svg,书看了一本 然后再找了个框架 snap.svg 摸索着写了个游戏给女儿玩。哈哈。即涨知识又娱乐。 转载于:https://www.cnblogs.com/goldli/p/7649898.html

idea .defaultMessage

idea .defaultMessage 等同于eclipse里的getDefaultMessage idea只能能获取到bean设置NotBlank等message响应到jsp的key&#xff0c;不能在jsp里获取validationMessage properties文件里的value 那么我们就不要value只要key 转载于:https://www.cnblogs.com/duneF/p/7654780.h…