03 | 你应该知道的Servlet规范和Servlet容器

浏览器发给服务端的是一个HTTP格式的请求,HTTP服务器收到这个请求后,需要调用服务端程序来处理,所谓的服务端程序就是你写的java类,一般来说不同的请求需要不同的java类来处理.

那么问题来了,HTTP服务器怎么知道要带哦用哪个java类的哪个方法呢?最直接的作法就是在HTTP服务器里写一堆if else逻辑判断:如果是A请求就调用X类的M1方法,如果是B请求就调用Y类的M2方法.但这样做明显由问题,因为HTTP服务器的代码跟业务逻辑耦合在一起了,如果新加一个业务方法还要修改HTTP服务器的代码.

那该怎么解决这个问题呢?我们知道,面向接口编程是解决耦合的法宝,于是又一伙人就定义了一个接口,各种业务类都必须实现这个接口,这个接口就叫Servlet接口,有时我们也把实现了Servlet接口的业务类叫Servlet.

但是这里还有一个问题,对于特定的请求,HTTP服务器如何知道由哪个Servlet来处理呢?Servlet又是由谁来实例化呢?显然HTTP服务器不适合这个工作,否则又和业务类耦合了.

于是,还是那伙人又发明了Servlei容器,Servlet容器用来加载和管理业务类.HTTP服务器不直接跟业务类打交道,而是把请求交给Servlet容器去处理,Servlet容器会将请求转发到具体的Servlet,如果这个Servlet还没创建,就加载并实例化这个Servlet,然后调用这个Servlet的接口方法,因此Servlet接口其实是Servlet容器跟业务类之间的接口,下面我们通过一张图来加深理解.


图的左边表示HTTP服务器直接调用具体的业务类,他们是紧耦合的.再看图的右边,HTTP服务器不直接调用业务类,而是把请求交给容器处理,容器通过Servlet接口调用业务类.因此Servlet接口和Servlet容器的出现,达到了HTTP服务器与业务类解耦的目的.

而Servlet接口和Servlet容器这一整套规范叫做Servlet规范.Tomcat和Jetty都按照Servlet规范的要求实现了Servlet容器,同时他们也具有HTTP服务器的功能,作为Java程序员,如果我们要实现新的业务功能,只需要实现一个Servlet,并把它注册到Tomcat(Servlet容器) 中,剩下的事情就由Tomcat帮我们处理了.

接下来我们看看Servlet接口具体是怎么定义的,以及Servlet规范又有哪些要重点关注的地方呢?

Servlet接口

Servlet 接口定义了下面五个方法:

public interface Servlet {

    void init(ServletConfig config) throws ServletException;


    ServletConfig getServletConfig();


    void service(ServletRequest req, ServletResponse res)throws ServletException, IOException;


    String getServletInfo();


    void destroy();

}

其中最重要的是service方法,具体业务类在这个方法里实现处理逻辑.这个方法有两个参数:ServletRequest和ServletResponse,ServletRequest用来封装请求信息,ServletResponse用来封装响应信息,因此本质上这两个类是对通信协议的封装.

比如HTTP协议中的请求和响应就是对应了HttpServletRequest和HttpServletResponse这两个类.你可以通过HttpServletRequest来获取所有请求相关的信息,包括请求路径,Cookie,HTTP头,请求参数等,此外,我们还可以通过HttpServletRequest来创建和获取Session,而HttpServletResponse是用来封装HTTP响应的.

你可以看到接口中还有两个跟声明周期有关的方法init和destory,这是一个比较贴心的设计,Servlet容器在加载Servlet类的时候会调用init方法,在卸载的时候会调用destory方法,我们可能会在init方法里初始化一些资源,并在destory方法里释放这些资源,比如Spring MVC中的DispatcherServlet,就是在init方法里创建了自己的Spring容器.

你还会注意到ServletConfig这个类,ServletConfig的作用就是封装Servlet的初始化参数,你可以在web.xml给Servlet参数,并在程序里通过getServletConfig方法拿到这些参数,

我们知道,有接口一般就有抽象类,抽象类用来实现接口和封装通用的逻辑,因此Servlet规范提供了GenericServlet抽象类,我们可以通过扩展它来实现Servlet.虽然Servlet规范并不在乎通信协议是什么,但是大多数的Servlet都是在HTTP环境中处理的,因此Servlet规范还提供了HttpServlet来继承GenericServlet,并且加入了HTTP特性,这样我们通过继承HttpServlet类来实现自己的Servlet,只需要重写两个方法:doGet和doPost.

Servlet容器

前面提到,为了解耦,HTTP服务器不直接调用Servlet,而是把请求交给Servlet容器来处理,那Servlet容器又是怎么工作的呢?接下来我会介绍Servlet容器大体的工作流程,一起来聊聊我们关心的两个话题:Web应用的目录格式是什么样的,以及我该怎样扩展和定制化Servlet容器的功能.

工作流程

当客户请求某个资源时,HTTP服务器会用一个ServletRequest对象把客户的请求信息封装起来,然后调用Servlet容器的service方法,Servlet容器拿到请求后,根据请求的URL和Servlet的映射关系,找到相应的Servlet,如果Servlet还没有被加载,就用反射机制创建这个Servlet,并调用Servlet的init方法来完成初始化,接着调用Servlet的service方法来处理请求,把ServletResponse对象返回给HTTP服务器,HTTP服务器会把响应发送给客户端,同样我通过一张图来帮助你理解;

Web应用

Servlet容器会实例化和调用Servlet,那Servlet是怎么注册到Servlet容器中的呢?一般来说,我们是以Web应用程序的方式来部署Servlet的,而根据Servlet规范,Web应用程序有一定的目录结构,在这个目录下分别放置了Servlet的类文件,配置文件以及静态资源,Servlet容器通过读取配置文件,就等找到并加载Servlet.Web应用的目录结构大概是下面这样:

| - MyWebApp

      | -  WEB-INF/web.xml        -- 配置文件,用来配置 Servlet 等

      | -  WEB-INF/lib/          -- 存放 Web 应用所需各种 JAR 包

      | -  WEB-INF/classes/      -- 存放你的应用类,比如 Servlet 类

      | -  META-INF/              -- 目录存放工程的一些信息

Servlet规范里定义了ServletContext这个接口来对应一个Web应用,Web应用部署好后,Servlet容器在启动时会加载Web应用,并为每个Web应用创建唯一的ServletContext对象.你可以把ServletContext堪称时一个全局对象,一个Web应用可能有多个Servlet,这些Servlet可以通过全局的ServletContext来共享数据,这些数据包括Web应用的初始化参数,Web应用目录下的文件资源等,由于Servlet Context持有所有Servlet实例,你还可以通过它来实现Servlet请求的转发.

扩展机制

不知道你有没有发现,引入了Servlet规范后,你不需要关心Socket网络通信,不需要关心HTTP协议,也不需要关心你的业务类时如何被实例化和调用的.因为这些都被Servlet规范标准化了,你只关心怎么实现你的业务逻辑,这对程序员来说是件好事,但也有不方便的一面,所谓规范就是说大家都要遵守,就会千篇一律,但是如果这个规范不能满足你的业务的个性化需求,就有问题了,因此设计一个规范或者一个中间件,要充分考虑可扩展性,Servlet规范提供了两种扩展机制:Filter和Listener.

Filter是过滤器,这个接口允许你对请求和响应做一些统一的定制化处理,比如你可以根据请求的频率来限制访问,或者根据国家地区的不同来修改响应内容,过滤器的工作原理是这样的:Web应用部署完成后,Servlet容器需要实例化Filter并把Filter链接成一个FilterChain,当请求进来时,获取第一个Filter调用doFilter方法,doFilter方法负责调用这个FilterChain中的下一个Filter.

Listener是监听器,这是另一种扩展机制 ,当web应用在Servlet容器中运行时,Servlet容器内部会不断的发生各种事件,如web应用的启动和停止,用户请求到达等,Servlet容器提供了一些默认的监听器来监听这些事件,当事件发生时,Servlet容器会负责调用监听器的方法.当然,你可以定义自己的监听器去监听你感兴趣的事件,将监听器配置在web.xml中,比如Spring就实现了自己的监听器,来监听ServletContext的启动事件,目的是当Servlet容器启动时,创建并初始化全局的Spring容器.

本期精华

今天我们学习了什么是Servlet,回顾一下,Servlet本质上是一个接口,实现了Servlet接口的业务类也叫Servlet.Servlet接口其实是Servlet容器跟具体Servlet业务类之间的接口,Servlet接口跟Servlet容器这整套规范叫做Servlet规范,而Servlet规范使得程序员可以专注业务逻辑的开发,同时Servlet规范也给开发者提供了扩展的机制Filter和Listener.

Filter是干预过程的,它是过程的一部分,是基于过程行为的.

Listener是基于状态的,任何行为改变同一个状态,触发的事件是一致的.

©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 160,026评论 4 364
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 67,655评论 1 296
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 109,726评论 0 244
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 44,204评论 0 213
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 52,558评论 3 287
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 40,731评论 1 222
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 31,944评论 2 314
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 30,698评论 0 203
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 34,438评论 1 246
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 30,633评论 2 247
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 32,125评论 1 260
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 28,444评论 3 255
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 33,137评论 3 238
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 26,103评论 0 8
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 26,888评论 0 197
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 35,772评论 2 276
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 35,669评论 2 271

推荐阅读更多精彩内容

  • Based on Java™ Servlet Specification v3.1 [TOC] Servlet和S...
    0x70e8阅读 1,261评论 0 7
  • Servlet接口 Servlet规范的核心接口即是Servlet接口,它是所有Servlet类必须实现的接口,在...
    java日记阅读 1,792评论 0 2
  • IOC 控制反转容器控制程序对象之间的关系,而不是传统实现中,有程序代码之间控制,又名依赖注入。All 类的创建,...
    irckwk1阅读 868评论 0 0
  • 这部分主要是与Java Web和Web Service相关的面试题。 96、阐述Servlet和CGI的区别? 答...
    杂货铺老板阅读 1,344评论 0 10
  • 转自陈明乾的博客,可能有一定更新。 转原文声明:原创作品,允许转载,转载时请务必以超链接形式标明文章 原始出处 、...
    C86guli阅读 4,603评论 6 72