tomcat原理解析(一):一个简单的实现
一 概述
前段时间去面试,被人问到了tomcat实现原理。由于平时没怎么关注容器的实现细节,这个问题基本没回答上来。所以最近花了很多时间一直在网上找资料和看tomcat的源码来研究里面处理一个HTTP请求的流程。网上讲tomcat的帖子比较多,大多都是直接切入主题看其源码,从我个人感受来说直接研究其源码实现比较难理解和非常枯燥,需要由简到难,慢慢深入。
二 一个简单tomcat服务器实现
tomat是一个servlet容器,来处理http请求。在平时的使用中我们都会再浏览器中输入http地址来访问服务资源,比如格式http://host[":"port][abs_path]。从浏览器到服务端的一次请求都遵循http协议,在网络上其实走仍然是tcp协议,即我们常使用的socket来处理客户端和服务器的交互。根据输入的http地址可以知道服务器的IP地址和端口,根据这两个参数就可以定位到服务器的唯一地址。tomcat根据http地址端口后面的资源路径就可以知道反馈什么样的资源给浏览器。下面给出了一个非常简单的代码模拟了tomcat的简单实现
- package com;
- import java.io.*;
- import java.net.ServerSocket;
- import java.net.Socket;
- import java.net.URLDecoder;
- import java.util.StringTokenizer;
- public class TomcatServer {
- private final static int PORT = 8080;
- public static void main(String[] args) {
- try {
- ServerSocket server = new ServerSocket(PORT);//根据端口号启动一个serverSocket
- ServletHandler servletHandler=new ServletHandler(server);
- servletHandler.start();
- } catch (Exception e) {
- e.printStackTrace();
- }
- }
- private static class ServletHandler extends Thread{
- ServerSocket server=null;
- public ServletHandler(ServerSocket server){
- this.server=server;
- }
- @Override
- public void run() {
- while (true) {
- try {
- Socket client = null;
- client = server.accept();//ServerSocket阻塞等待客户端请求数据
- if (client != null) {
- try {
- System.out.println("接收到一个客户端的请求");
- //根据客户端的Socket对象获取输入流对象。
- //封装字节流到字符流
- BufferedReader reader = new BufferedReader(new InputStreamReader(client.getInputStream()));
- // GET /test.jpg /HTTP1.1
- //http请求由三部分组成,分别是:请求行、消息报头、请求正文。
- //这里取的第一行数据就是请求行。http协议详解可以参考http://www.cnblogs.com/li0803/archive/2008/11/03/1324746.html说的很详细
- String line = reader.readLine();
- System.out.println("line: " + line);
- //拆分http请求路径,取http需要请求的资源完整路径
- String resource = line.substring(line.indexOf('/'),line.lastIndexOf('/') - 5);
- System.out.println("the resource you request is: "+ resource);
- resource = URLDecoder.decode(resource, "UTF-8");
- //获取到这次请求的方法类型,比如get或post请求
- String method = new StringTokenizer(line).nextElement().toString();
- System.out.println("the request method you send is: "+ method);
- //继续循环读取浏览器客户端发出的一行一行的数据
- while ((line = reader.readLine()) != null) {
- if (line.equals("")) {//当line等于空行的时候标志Header消息结束
- break;
- }
- System.out.println("the Http Header is : " + line);
- }
- //如果是POST的请求,直接打印POST提交上来的数据
- if ("post".equals(method.toLowerCase())) {
- System.out.println("the post request body is: "
- + reader.readLine());
- }else if("get".equals(method.toLowerCase())){
- //判断是get类型的http请求处理
- //根据http请求的资源后缀名来确定返回数据
- //比如下载一个图片文件,我这里直接给定一个图片路径来模拟下载的情况
- if (resource.endsWith(".jpg")) {
- transferFileHandle("d://123.jpg", client);
- closeSocket(client);
- continue;
- } else {
- //直接返回一个网页数据
- //其实就是将html的代码以字节流的形式写到IO中反馈给客户端浏览器。
- //浏览器会根据http报文“Content-Type”来知道反馈给浏览器的数据是什么格式的,并进行什么样的处理
- PrintStream writer = new PrintStream(client.getOutputStream(), true);
- writer.println("HTTP/1.0 200 OK");// 返回应答消息,并结束应答
- writer.println("Content-Type:text/html;charset=utf-8");
- writer.println();
- //writer.println("Content-Length:" + html.getBytes().length);// 返回内容字节数
- writer.println("<html><body>");
- writer.println("<a href='www.baidu.com'>百度</a>");
- writer.println("<img src='https://ss0.bdstatic.com/5aV1bjqh_Q23odCf/static/superman/img/logo/bd_logo1_31bdc765.png'></img>");
- writer.println("</html></body>");
- //writer.println("HTTP/1.0 404 Not found");// 返回应答消息,并结束应答
- writer.println();// 根据 HTTP 协议, 空行将结束头信息
- writer.close();
- closeSocket(client);//请求资源处理完毕,关闭socket链接
- continue;
- }
- }
- } catch (Exception e) {
- System.out.println("HTTP服务器错误:"
- + e.getLocalizedMessage());
- }
- }
- } catch (Exception e) {
- e.printStackTrace();
- }
- }
- }
- private void closeSocket(Socket socket) {
- try {
- socket.close();
- } catch (IOException ex) {
- ex.printStackTrace();
- }
- System.out.println(socket + "离开了HTTP服务器");
- }
- private void transferFileHandle(String path, Socket client) {
- File fileToSend = new File(path);
- if (fileToSend.exists() && !fileToSend.isDirectory()) {
- try {
- //根据Socket获取输出流对象,将访问的资源数据写入到输出流中
- PrintStream writer = new PrintStream(client.getOutputStream());
- writer.println("HTTP/1.0 200 OK");// 返回应答消息,并结束应答
- writer.println("Content-Type:application/binary");
- writer.println("Content-Length:" + fileToSend.length());// 返回内容字节数
- writer.println();// 根据 HTTP 协议, 空行将结束头信息
- FileInputStream fis = new FileInputStream(fileToSend);
- byte[] buf = new byte[fis.available()];
- fis.read(buf);
- writer.write(buf);
- writer.close();
- fis.close();
- } catch (IOException e) {
- e.printStackTrace();
- }
- }
- }
- }
- }
三 实践
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源码。
tomcat原理解析(一):一个简单的实现的更多相关文章
- Skinned Mesh原理解析和一个最简单的实现示例
Skinned Mesh 原理解析和一个最简单的实现示例 作者:n5 Email: happyfirecn##yahoo.com.cn Blog: http://blog.csdn.net/n5 ...
- 《深度解析Tomcat》 第一章 一个简单的Web服务器
本章介绍Java Web服务器是如何运行的.从中可以知道Tomcat是如何工作的. 基于Java的Web服务器会使用java.net.Socket类和java.net.ServerSocket类这两个 ...
- tomcat原理解析(二):整体架构
一 整体结构 前面tomcat实现原理(一)里面描述了整个tomcat接受一个http请求的简单处理,这里面我们讲下整个tomcat的架构,以便对整体结构有宏观的了解.tomat里面由很多个容器结合在 ...
- 窥探原理:实现一个简单的前端代码打包器 Roid
roid roid 是一个极其简单的打包软件,使用 node.js 开发而成,看完本文,你可以实现一个非常简单的,但是又有实际用途的前端代码打包工具. 如果不想看教程,直接看代码的(全部注释):点击地 ...
- git原理学习记录:从基本指令到背后原理,实现一个简单的git
一开始我还担心 git 的原理会不会很难懂,但在阅读了官方文档后我发现其实并不难懂,似乎可以动手实现一个简单的 git,于是就有了下面这篇学习记录. 本文的叙述思路参照了官方文档Book的原理介绍部分 ...
- how tomcat works 札记(两)----------一个简单的servlet集装箱
app1 (看着眼前这章建议读者,看how tomcat works 札记(一个)----------一个简单的webserver http://blog.csdn.net/dlf123321/art ...
- 【第十一课】Tomcat原理解析【转】
目录 一.Tomcat顶层架构 二.Tomcat顶层架构小结: 三.Connector和Container的微妙关系 四.Connector架构分析 五.Container架构分析 六.Contain ...
- Tomcat剖析(二):一个简单的Servlet服务器
Tomcat剖析(二):一个简单的Servlet服务器 1. Tomcat剖析(一):一个简单的Web服务器 2. Tomcat剖析(二):一个简单的Servlet服务器 3. Tomcat剖析(三) ...
- Tomcat剖析(一):一个简单的Web服务器
Tomcat剖析(一):一个简单的Web服务器 1. Tomcat剖析(一):一个简单的Web服务器 2. Tomcat剖析(二):一个简单的Servlet服务器 3. Tomcat剖析(三):连接器 ...
随机推荐
- pymysql模块操作数据库
pymysql模块是python操作数据库的一个模块 connect()创建数据库链接,参数是连接数据库需要的连接参数 使用方式: 模块名称.connect() 参数: host=数据库ip po ...
- LOJ2538. 「PKUWC2018」Slay the Spire【组合数学】
LINK 思路 首先因为式子后面把方案数乘上了 所以其实只用输出所有方案的攻击力总和 然后很显然可以用强化牌就尽量用 因为每次强化至少把下面的牌翻一倍,肯定是更优的 然后就只有两种情况 强化牌数量少于 ...
- .NET 中小心嵌套等待的 Task,它可能会耗尽你线程池的现有资源,出现类似死锁的情况
一个简单的 Task 不会消耗多少时间,但如果你不合适地将 Task 转为同步等待,那么也可能很快耗尽线程池的所有资源,出现类似死锁的情况. 本文将以一个最简单的例子说明如何出现以及避免这样的问题. ...
- javascript : location 对象
window.location: window的location对象 window.location.href 整个URl字符串(在浏览器中就是完整的地址栏) window.location.prot ...
- java1.8操作日期
java1.8获取年份: int year = Calendar.getInstance().get(Calendar.YEAR); StringBuilder code = new StringBu ...
- CSS3 Flexbox轻巧实现元素的水平居中和垂直居中(转)
CSS3 Flexbox轻松实现元素的水平居中和垂直居中 网上有很多关于Flex的教程,对于Flex的叫法也不一,有的叫Flexbox,有的叫Flex,其实这两种叫法都没有错,只是Flexbox旧一点 ...
- FastAdmin 浏览器 JS CSS 缓存如何更新?
由于代码修改,但文件名没有修改,因为浏览器对 JS 和 CSS 是缓存的,而且由于服务器无法控制客户端的缓存. 但是可以对 JS 和 CSS 的请求加上版本号,达到更新缓存的效果.
- mac系统下 Homebrew 使用
brew 又叫 Homebrew,是一款Mac OS平台下的软件包管理工具. brew 常用命令: 命令 作用 brew install [package] 安装包 brew uninstall [p ...
- SQL Server Reporting Service 报错:报表服务器无法解密用于访问报表服务器数据库中的敏感数据或加密数据的对称密钥,必须还原备份密钥或删除所有加密的内容。
出现这个问题,可以通过reporting services 配置管理工具来处理 首先,打开配置管理工具,连接. 在左侧的导航选项中选择Encryption Keys,将出现如图所示的界面,在右侧点击d ...
- 跟老齐学Django 项目实战笔记
创建项目 mysite 创建应用 blog mysit/settings.py配置app INSTALLED_APPS = [ 'django.contrib.admin', 'django.cont ...