详解Servlet

什么是Servlet

    Servlet是Server+Applet的缩写,表示一个服务应用。其实Servlet就是一套规范,我们按照这套规范写的代码就可以直接在Java的服务器上面运行。

Servlet接口

    既然Servlet是一套规范,那么最重要的当然就是接口了。Servlet3.1中Servlet的接口,如下代码清单:

servlet

        ▪️init方法在容器启动时被容器调用(当load-on-startup设置为负数或者不设置时会在Servlet第一次用到时才被调用),只会调用一次;

        ▪️getServletConfig方法用于获取ServletConfig,;

        ▪️service方法用于具体处理一个请求;

        ▪️getServletInfo方法可以获取一些Servlet相关的信息,如作者、版权等,这个方法需要自己实现,默认返回空字符串;

        ▪️destroy方法主要用于在Servlet销毁(一般指关闭服务器)时释放一些资源,也只会调用一次。

    init方法被调用时会接收到一个ServletConfig类型的参数,是容器传进去的。ServletConfig顾名思义指的是Servlet的配置,我们在web.xml中定义Servlet时通过init-param标签配置的参数就是通过ServletConfig来保存的,比如,定义Spring MVC的Servlet时指定配置文件位置的contextConfigLocation参数就保存在ServletConfig中,例如下面的配置:

web.xml

    Tomcat中Servlet的init方法是在org.apache.catalina.core.StandardWrapper的initServlet方法中调用的,ServletConfig传入的是StandardWrapper(里面封装着Servlet)自身的门面类StandardWrapperFacade。其实这个也很容易,Servlet是通过xml文件配置的,在解析xml时就会把配置参数給设置进去,这样StandardWrapper本身就包含配置项了,当然并不是StandardWrapper的所有内容都是Config相关的,所以就用了其门面Facade类。下面是ServletConfig接口的定义:

ServletConfig接口

    getServletName用于获取Servlet的名字,也就是我们在web.xml中定义的servlet-name;getInitParameter方法用于获取init-param配置的参数;getInitParameterNames用于获取配置的所有init-param的名字集合;getServletContext非常重要,它的返回值ServletContext代表的是我们这个应用本身,如果你看了前面Tomcat的分析就会想到,ServletContext其实就是Tomcat中Context的门面类AppliactionContextFacade(具体代码参考StandardContext的getServletContext)。既然ServletContext代表应用本身,那么ServletContext里面设置的参数就可以被当前应用的所有Servlet共享了。我们做项目的时候都知道参数可以保存在Session中,也可以保存在Application中,而后者很多时候就是保存在了ServletContext中。

    我们可以这么理解,ServletConfig是Servlet级的,而ServletContext是Context(也就是Application级的)。当然,ServletContext的功能要强大很多,并不只是保存一下配置参数,否则就叫ServletContextConfig了。

ServletContext接口

    有的读者可能会想,Servlet级和Context级都可以操作,那有没有更高一层的站点也就是Tomcat中的Host级的相应操作呢?在Servlet的标准例其实还真有,在ServetContext接口中有这么一个方法:public ServletContext getContext(String uripath),它可以根据路径获取到同一个站点下单别的应用等ServletContext!当然由于安全的原因,一般会返回null,如果想使用需要进行一些设置。

    ServletConfig和ServletContext最常见的使用之一是传递初始化参数。我们就以spring配置中使用得最多的contextConfigLocation参数为例来看一下:

web.xml

注意:如果设置metadata-complete="true",会在启动时不扫描注解(annotation)。如果不扫描注解的话,用注解进行的配置就无法生效,例如:@WebServlet

    上面通过context-param配置的contextConfigLocation配置到了ServletContext中,而通过servlet下单init-param配置的contextConfigLocation配置到了ServletConfig中。在Servlet中可以分别通过它们的getInitParameter方法进行获取,比如:

String contextLocation = getServletConfig.getInitParameter("contextConfigLocation"); 


String contextLocation = getServletConfig().getServletContext().getInitParameter("contextConfigLocation");

    为了操作方便,GenericServlet定义了getInitParameter方法,内部返回getServletConfig().getInitParameter的返回值,因此我们如果需要获取ServletConfig中的参数,可以不再调用getServletConfig(),而直接调用getInitParameter。

    另外ServletContext中非常常用的用法就是保存Application级的属性,这个可以使用setAttribute来完成,比如:

getServltContext().setAttribute("contextConfigLocation","new path");

    需要注意的是,这里设置的同名Attribute并不会覆盖initParameter中的参数值,它们是两套数据,互不干扰。ServletConfig不可以设置属性。

GenericServlet

GenericServlet是Servlet的默认实现,主要做了三件事:

    ▪️实现了ServletConfig接口,我们可以直接调用ServletConfig里面的方法;

    ▪️提供了无参的init方法;

    ▪️提供了log方法。

    GenericServlet实现了ServletConfig接口,我们在需要调用ServletConfig中方法等时候可以直接调用,而不再需要先获取ServletConfig了,比如,获取ServletContext的时候可以直接调用getServletContext,而无须调用getServletConfig.getServletContext()了,不过其底层实现其实是在内部调用了。getServletContext的代码如下:

javax.servlet.GenericServlet                                           

    GenericServlet实现了Servlet的init(ServletConfig config)方法,在里面将config设置給了内部变量config,然后调用了无参点init()方法,这个方法是个模版方法,在子类中可以通过通过覆盖它来完成自己的初始化工作,代码如下:

javax.servlet.GenericServlet

这种做法有三个作用:

    ▪️首先,将参数config设置給了内部属性config,这样就可以在ServletConfig的接口方法中直接调用config的相应方法来执行;

    ▪️其次,这么做之后我们在写Servlet的时候就可以只处理自己的初始化逻辑,而不需要再关心config了;

    ▪️还有一个作用就是在重写init()方法时也不需要再调用super.init(config)了。如果在自己的Servlet中重写了带参数的init方法,那么一定要记着调用super.init(config),否则这里的config属性就接收不到值,相应地ServletConfig接口方法也就不能执行了。

    GenericServlet提供了2个log方法,一个记录日志,一个记录异常。具体实现是通过传给ServletContext的日志实现的。

javax.servlet.GenericServlet

    一般我们都有自己的日志处理方式,所以这个用得部署很多。

    GenericServlet是与具体协议无关的。

HttpServlet

    HttpServlet是用HTTP协议实现的Servlet的基类,写Servlet时直接继承它就可以了,不需要再从头实现Servlet接口,我们要分析的Spring MVC中的DispatcherServlet就是继承的HttpServlet。既然HttpServlet是跟协议相关的,当然主要关心的是如何处理请求了,所以HttpServlet主要重写了service方法。在service方法中首先将ServletReuqest和ServletResponse转换为了HttpServletRequest和HttpServletResponse,然后根据Http请求的类型不同将请求路由到了的处理方法。代码如下:

javax.servlet.http.HttpServlet

service(request,response)的具体实现,如下代码清单所示:

service(request,response)的具体实现

承接上图

承接上图

    具体处理方法是doXXX的结构,如最常用doGet、doPost就是在这里定义的。doGet、doPost、doPut和doDelet方法都是模版方法,而且如果子类没有实现将抛出异常,在调用doGet的请求,然后返回空body的Response;doOptions和doTrace正常不需要使用,主要是用来做一些调试工作,doOptions返回所有支持的处理类型的集合,正常情况下可以禁用,doTrace是用来远程诊断服务器的,它会将接收到的header原封不动地返回,这种做法很可能会被黑客利用,存在安全漏洞,所以如果不是必须使用,最好禁用。由于doOptions和doTrace的功能非常固定,所以HttpServlet做了默认的实现。doGet代码如下(doPost、doPut、doDelete与之类似);

javax.servlet.http.HttpServlet

    这就是HttpServlet,它主要将不同的请求方式路由到了不同的处理方法。不过Spring MVC中由于处理思路不一样,又将所有请求合并到了统一的一个方法进行处理,在Spring MVC中再详细讲解。

小知识

▪️HttpServlet继承自GenericServlet,如下代码清单:

HttpServlet

▪️GenericServlet实现了Servlet,ServletConfig接口,如下代码清单

GenericServlet

▪️Spring MVC与HttpServlet之间的关系,如下代码清单

spring-webmvc-4.3.9.RELEASE-----HttpServletBean

不出所料,但是HttpServletBean是个抽象类,看看FrameworkServlet怎么回事,如下代码清单

FrameworkServlet

然而FrameworkServlet仍然是个抽象类,我们熟悉的DispatcherServlet上场了。

DispatcherServlet

    spring MVC中servlet一共有三个层次,分别是HttpServletBean、FrameworkServlet和DispatcherServlet。HttpServletBean直接继承自java的HttpServlet,其作用是将Servlet中配置的参数设置到相应的属性;FrameworkServlet继承HttpServletBean,有processRequest方法,主要作用是初始化了WebApplicationContext,DispactherServlet继承了FrameworkServlet,重写了doService方法,主要作用初始化了自身的9个组件。

     DispactherServlet中的核心方法doService和doDispatch

    doService:重写了FrameworkServlet的doService方法,主要作用是,检查是否为include请求,假如是,对request的attribute进行snapshot,检查是否有FlashMap,设置request一些属性,然后进入doDispatch方法;

    doDispatch:检查是否为上传请求(Multipart),根据request生成HandlerMapping,找到Handler和Interceptor,根据Handler找到HandlerAdapter,Adapter会检查Last-Modified,执行相应的Interceptor;

    整体结构非常简单--分三个层次做了三件事,但具体实现过程还是有点复杂的。这其实也是spring的特点:结构简单,实现复杂。结构简单主要是顶层设计好,实现复杂的主要是提供的功能比较多,可配置的地方也非常多个九个组件分别是:

    ▪️HandlerMapping

    ▪️HandlerAdapter

    ▪️HandlerExceptionResolver

    ▪️ViewResovler

    ▪️RequestToViewNameTranslator

    ▪️LocaleResolver

    ▪️ThemeResolver

    ▪️MultipartResolver

    ▪️FlashMapManager

《详解Spring MVC:上》

《详解Spring MVC:下》

**如果需要給我修改意见的发送邮箱:erghjmncq6643981@163.com**

**资料参考:《看透Spring MVC-源代码分析与实践》**

**转发博客,请注明,谢谢。**

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

推荐阅读更多精彩内容