tomcat原理分析与简单实现
tomcat原理分析与简单实现
https://blog.csdn.net/u014795347/article/details/52328221
1.tomcat实际是运行在jvm中的一个进程。我们把它定义为【中间件】,顾名思义,他是一个在java项目与jvm之间的
中间容器。我们的web项目没有入口方法(main方法),那么他是如何运行起来并为客户端返回数据的呢?
2.web项目[就javaee而讲]的本质,是一大堆的资源文件和方法。其中没有main方法,意味着web项目中的方法不会自动
运行起来。
3.这样,我们想想也知道,我们把web项目部署进tomcat的webapp中的目的是很明确的,那就是希望tomcat去调用我们
写好的方法去为客户端返回需要的资源和数据。
4.tomcat可以运行起来,并调用我们写好的方法。那么,tomcat一定有一个main方法。
5.对于tomcat而言,它并不知道我们会有什么样的方法,这些都只是在项目被部署进webapp下后才确定的,由此分析,
必然用到了java的反射来实现类的动态加载、实例化、获取方法、调用方法。
6.那么tomcat如何确定调用什么方法呢。这取却于客户端的请求,举个栗子:
【http://127.0.0.1:8080/a/b/c.htm?a=1&b=2】这样的一个请求,通过http协议,使用GET方法在浏览器发往本机的8080端口,
携带的参数包含两部分a.方法,包含此方法的路径【/a/b/c.htm】,这里的方法为c,以htm标注,/a/b代表路径
这样可以允许在不同路径下存在同名方法,更可以唯一定位一个方法。b.参数,包含参数名和参数值【a=1&b=2】
通过这样的方法,要调用哪个方法,以及需要什么参数,我们的tomcat一目了然。
7.综上所述,我们有下面的总结:
a.tomcat需要main方法启动。
b.tomcat需要监听本机上的某个端口。
c.tomcat需要抓取此端口上来自客户端的链接并获得请求调用的方法与参数。
d.tomcat需要根据请求调用的方法,动态地加载方法所在的类,完成累的实例化并通过该实例获得需要的方法最终将请求
传入方法执行。
e.将结果返回给客户端(jsp/html页面、json/xml字符串)。
二、模型实现
1.Main.java
- package cn.wwyxxmiemie.littletomcat;
- import java.io.BufferedReader;
- import java.io.IOException;
- import java.io.InputStreamReader;
- import java.lang.reflect.Constructor;
- import java.lang.reflect.InvocationTargetException;
- import java.lang.reflect.Method;
- import java.net.ServerSocket;
- import java.net.Socket;
- import cn.wwyxxmiemie.littletomcat.exclass.ExClass;
- import cn.wwyxxmiemie.littletomcat.util.ClintRequestBean;
- /**
- * 这是littletomcat的类,是整个容器的入口类
- * 程序在这个类的main方法启动
- * @author 卫卫羊习习
- */
- public class Main {
- /**
- * 容器主方法
- * @param args
- * @throws InstantiationException
- * @throws ClassNotFoundException
- * @throws SecurityException
- * @throws NoSuchMethodException
- */
- public static void main(String[] args) throws InstantiationException {
- System.out.println("little_tomcat_is_running!");
- try {
- ServerSocket serverSocket = new ServerSocket(80);
- while (true) {
- //服务器每接受一次请求,创建一个socket对象
- Socket socket = serverSocket.accept();
- BufferedReader bReader = new BufferedReader(new InputStreamReader(socket.getInputStream()));
- String line = bReader.readLine();
- if (!(null==line)) {
- ClintRequestBean requestBean = new ClintRequestBean(line);
- System.out.println("客户端请求:"+requestBean.toReadString());
- System.out.println("请求参数[路径]:"+requestBean.getRequestParm().get("path"));
- System.out.println("请求参数[参数表]:"+requestBean.getRequestParm().get("attrs"));
- ClassLoader classLoader = ClassLoader.getSystemClassLoader();
- try {
- classLoader.loadClass("cn.wwyxxmiemie.littletomcat.exclass.ExClass");
- System.out.println("动态加载ExClass类--成功");
- } catch (ClassNotFoundException e) {
- e.printStackTrace();
- System.out.println("动态加载ExClass类--失败");
- }
- Class<?> exClass = null;
- try {
- exClass = Class.forName("cn.wwyxxmiemie.littletomcat.exclass.ExClass");
- System.out.println("动态初始化ExClass类--成功");
- } catch (ClassNotFoundException e) {
- e.printStackTrace();
- System.out.println("动态初始化ExClass类--失败");
- }
- Method method;
- try {
- method = exClass.getMethod("test", null);
- System.out.println("得到ExClass对象的"+method.getName()+"方法");
- try {
- System.out.println("执行ExClass对象的"+method.getName()+"方法");
- method.invoke(exClass.newInstance(), null);
- } catch (IllegalAccessException e) {
- // TODO Auto-generated catch block
- e.printStackTrace();
- } catch (IllegalArgumentException e) {
- // TODO Auto-generated catch block
- e.printStackTrace();
- } catch (InvocationTargetException e) {
- // TODO Auto-generated catch block
- e.printStackTrace();
- }
- } catch (NoSuchMethodException e) {
- // TODO Auto-generated catch block
- e.printStackTrace();
- } catch (SecurityException e) {
- // TODO Auto-generated catch block
- e.printStackTrace();
- }
- }
- bReader.close();
- socket.close();
- }
- } catch (IOException e) {
- e.printStackTrace();
- }
- }
- }
2.ClintRequestBean.Java
- package cn.wwyxxmiemie.littletomcat.util;
- import java.util.HashMap;
- import java.util.Map;
- /**
- * 客户端请求实体
- * 用于封装客户端的链接数据
- * @author 卫卫羊习习
- */
- public class ClintRequestBean {
- //以一个请求举例:http://127.0.0.1/www/qqq/eee
- private String protocol;//协议类型(eg:http)
- private String protocolVersion;//协议版本(eg:1.1)
- private String data;//请求数据(eg:/www/qqq/eee)
- private String method;//请求方法:(eg:GET)
- /**
- * 客户端请求实体构造方法
- * @param protocol 协议类型 (eg:http)
- * @param protocolVersion 协议版本 (eg:1.1)
- * @param data 请求数据 (eg:/www/qqq/eee)【必须以‘/’分隔】
- * @param method 请求方法 (eg:GET)
- */
- public ClintRequestBean(String protocol, String protocolVersion, String data, String method) {
- super();
- this.protocol = protocol;
- this.protocolVersion = protocolVersion;
- this.data = data;
- this.method = method;
- }
- /**
- * 客户端请求实体构造方法
- * @param request 请求链接,一般针对一条完整的http链接
- */
- public ClintRequestBean(String request){
- super();
- String [] requestString = request.split(" ");
- this.method = requestString[0];
- this.data = requestString[1];
- String [] proAndVer = requestString[2].split("/");
- this.protocol = proAndVer[0];
- this.protocolVersion = proAndVer[1];
- }
- /**
- * 转化为可读String用于分析请求
- * @return
- */
- public String toReadString(){
- return "ClintRequestBean [protocol=" + protocol + ", protocolVersion=" + protocolVersion + ", data=" + data
- + ", method=" + method + "]";
- }
- /**
- * 得到请求的参数
- * @return map[请求路径|参数map]
- */
- public Map<String, Object> getRequestParm(){
- Map<String,Object> map = new HashMap<>();
- String [] parms = data.split("\\?");
- map.put("path", parms[0]);
- Map<String, String> attrs = new HashMap<>();
- String[] kvs = parms[1].split("&");
- for (String string : kvs) {
- String [] kv = string.split("=");
- attrs.put(kv[0], kv[1]);
- }
- map.put("attrs", attrs);
- return map;
- }
- public String getProtocol() {
- return protocol;
- }
- public void setProtocol(String protocol) {
- this.protocol = protocol;
- }
- public String getProtocolVersion() {
- return protocolVersion;
- }
- public void setProtocolVersion(String protocolVersion) {
- this.protocolVersion = protocolVersion;
- }
- public String getData() {
- return data;
- }
- public void setData(String data) {
- this.data = data;
- }
- public String getMethod() {
- return method;
- }
- public void setMethod(String method) {
- this.method = method;
- }
- @Override
- public String toString() {
- return this.method+" "+this.data+" "+this.protocol+"/"+this.protocolVersion;
- }
- }
3.ExClass.java 用于编译生成ExClass.class文件,供littletomcat动态加载
- package cn.wwyxxmiemie.littletomcat.exclass;
- /**
- * 测试类,被主容器动态调用
- * @author 卫卫羊习习
- */
- public class ExClass {
- /**
- * 测试方法
- */
- public void test(){
- System.out.println("ExClass.test()方法被调用");
- }
- }
三、检测
在浏览器发出如下http请求
【http://127.0.0.1/qqq/www/eee/?a=1&b=2】
littletomcat反映如下
四、注
以上代码并没有实现选择方法,但类的加载实则参数为字符串,大家理解原理就好
tomcat原理分析与简单实现的更多相关文章
- Shiro框架 (原理分析与简单实现)
Shiro框架(原理分析与简单实现) 有兴趣的同学也可以阅读我之前分享的:Java权限管理(授权与认证)CRM权限管理 (PS : 这篇博客里面的实现方式没有使用框架,完全是手写的授权与认证,可以 ...
- vue2.0 双向绑定原理分析及简单实现
Vue用了有一段时间了,每当有人问到Vue双向绑定是怎么回事的时候,总是不能给大家解释的很清楚,正好最近有时间把它梳理一下,让自己理解的更清楚,下次有人问我的时候,可以侃侃而谈. 一.首先介绍Obje ...
- spring之mvc原理分析及简单模拟实现
在之前的一篇博客中已经简单的实现了spring的IOC和DI功能,本文将在之前的基础上实现mvc功能. 一 什么是MVC MVC简单的说就是一种软件实现的设计模式,将整个系统进行分层,M(model ...
- Xss漏洞原理分析及简单的讲解
感觉百度百科 针对XSS的讲解,挺不错的,转载一下~ XSS攻击全称跨站脚本攻击,是为不和层叠样式表(Cascading Style Sheets, CSS)的缩写混淆,故将跨站脚本攻击缩写为XS ...
- Tomcat源码分析——请求原理分析(下)
前言 本文继续讲解TOMCAT的请求原理分析,建议朋友们阅读本文时首先阅读过<TOMCAT源码分析——请求原理分析(上)>和<TOMCAT源码分析——请求原理分析(中)>.在& ...
- Tomcat源码分析——请求原理分析(中)
前言 在<TOMCAT源码分析——请求原理分析(上)>一文中已经介绍了关于Tomcat7.0处理请求前作的初始化和准备工作,请读者在阅读本文前确保掌握<TOMCAT源码分析——请求原 ...
- Tomcat源码分析——请求原理分析(上)
前言 谈起Tomcat的诞生,最早可以追溯到1995年.近20年来,Tomcat始终是使用最广泛的Web服务器,由于其使用Java语言开发,所以广为Java程序员所熟悉.很多人早期的J2EE项目,由程 ...
- tomcat原理解析(一):一个简单的实现
tomcat原理解析(一):一个简单的实现 https://blog.csdn.net/qiangcai/article/details/60583330 2017年03月07日 09:54:27 逆 ...
- Servlet过滤器介绍之原理分析
zhangjunhd 的BLOG 写留言去学院学习发消息 加友情链接进家园 加好友 博客统计信息 51CTO博客之星 用户名:zhangjunhd 文章数:110 评论数:858 访问量:19 ...
随机推荐
- 利用asynchttpclient开源项目来把数据提交给服务器
可以通过github去查找asynchttpclient,并下载源代码,并加载到自己的工程中. 1.利用get方法提交 2.利用post方法来提交
- C#/.NET 使用 CommandLineParser 来标准化地解析命令行
CommandLineParser 是一款用于解析命令行参数的 NuGet 包.你只需要关注你的业务,而命令行解析只需要极少量的配置代码. 本文将介绍如何使用 CommandLineParser 高效 ...
- 字符串匹配--扩展KMP模板
对于一个字符串 s 以及子串 t ,扩展KMP可以用来求 t 与 s 的每个子串的最长公共前缀 ext [ i ],当然,如果有某个 ext 值等于 t 串的长度 lent ,那么就说明从其对应的 i ...
- 彻底删除vscode及安装的插件和个人配置信息
1.卸载vscode应用软件(在控制面板里面找不到改软件,所以只能进入应用所在文件夹进行卸载) ## 此步骤虽然删掉了应用软件,但是此时重新安装会发现之前下载的插件和个人配置信息都还会重新加载出来,所 ...
- VisualSVN安装配置与使用
VisualSVN安装配置与使用 1. 所选服务器安装包:VisualSVN-Server-2.1.3.msi. 2. 客户端安装包:TortoiseSVN-1.6.2.16344-win32-s ...
- [NN] 对于BackPropagation(BP, 误差反向传播)的一些理解
本文大量参照 David E. Rumelhart, Geoffrey E. Hinton and Ronald J. Williams, Learning representation by bac ...
- tomcat源码阅读之Server和Service接口解析
tomcat中的服务器组件接口是Server接口,服务接口是Service,Server接口表示Catalina的整个servlet引擎,囊括了所有的组件,提供了一种优雅的方式来启动/关闭Catali ...
- 【转】每天一个linux命令(24):Linux文件类型与扩展名
原文网址:http://www.cnblogs.com/peida/archive/2012/11/22/2781912.html Linux文件类型和Linux文件的文件名所代表的意义是两个不同的概 ...
- sed 给文件每行末尾追加相同字符
给文件file1每行末尾追加字符 ; sed 's/$/;/' file1
- 基于DRL和TORCS的自动驾驶仿真系统——之环境配置
基于DRL和TORCS的自动驾驶仿真系统 --之环境配置 玩TORCS和DRL差不多有一整年了,开始的摸爬滚打都是不断碰壁过来的,近来在参与CMU的DRL10703课程学习和翻译志愿者工作,也将自己以 ...