《看透SpringMVC源代码分析与实践》学习笔记(持续更新……)

    本文是我自己在秋招复习时的读书笔记,整理的知识点,也是为了防止忘记,尊重劳动成果,转载注明出处哦!如果你也喜欢,那就点个小心心,文末赞赏一杯豆奶吧,嘻嘻。 让我们共同成长吧……


第一篇  网站基础知识

第1章  网站架构及其演变过程

1.1  软件的三大类型

        单机类型,CS类型,BS类型

1.2    基础架构并不简单

        BS结构网络传输的分解方式有两种:一种是标准的OSI参考模型;另一种是TCP/IP参考模型。

        OSI参考模型:物理层、数据链路层、网络层、传输层、会话层、表示层、应用层

        TCP/IP参考模型:网络接入层、网际互联层、传输层、应用层

        对于TCP/IP参考模型的4层的理解:

        BS结构中,网络接入层没有相应协议,网际互联层是IP协议,传输层是TCP协议,应用层是HTTP协议。还用到了DNS协议。

1.3  架构演变的起点

1.4  海量数据的解决方案

    1.4.1    缓存和页面静态化

        缓存使用方式:通过程序直接保存在内存中和使用缓存框架。程序直接操作主要是使用Map,尤其是ConcurrentHashMap,常用的缓存框架有Ehcache、Memchache和Redis。缓存主要适用于数据变化不是很频繁的情况,如果是定期失效的失效机制,实时性也不能要求太高。

        页面静态化是将程序最后生成的页面保存起来,页面静态化对数据量大、并发量高两大问题都有好处。页面静态化可以在程序中使用模板技术生成,例如Freemarker、Velocity.也可以使用缓存服务器再应用服务器的上一层缓存生成的页面,例如Nginx、Squid。

    1.4.2  数据库优化

        常见的数据库优化:表结构优化、SQL语句优化、分区和分表、索引优化、使用存储过程代替直接操作、合理使用冗余。

    1.4.3    分离活跃数据

    1.4.4    批量读取和延迟修改

    1.4.5    读写分离

    1.4.6  分布式数据库

    1.4.7    NoSQL和Hadoop    

1.5    高并发的解决方案

    应用和静态资源分离、页面缓存、集群与分布式、反向代理、CDN

1.6  底层的优化

1.7  小结



第2章    常见协议和标准

2.1  DNS协议

        DNS作用是将域名解析为IP.

2.2    TCP/IP协议与Socket

        TCP在传输之前会进行三次沟通,称作“三次握手”,传完数据断开时要进行四次沟通,称作“四次挥手”。TCP中的两个序号和三个标志位的含义:

        seq:表示传输数据的序号。ack:表示确认号。ACK:确认位。SYN:同部位。FIN:终止位。

2.3  HTTP协议

        HTTP中的报文分为:请求报文、响应报文。这两种类型都包括:首行、头部和主体。

HTTP报文结构

        请求报文中的方法指:GET、HEAD、POST、PUT、DELETE,响应报文中的状态码分为5类:

5种类型的状态码

2.4    Servlet与Java Web开发

        Servlet是J2EE标准的一部分,是JavaWeb开发的标准。Servlet的作用是对接收到的数据进行处理并生成要返回给客户端的结果。要想使用Servlet需要相应的Servlet容器,常见的Servlet容器时Tomcat.

第3章    DNS的设置

第4章    Java中Socket的用法

    本章介绍Java中的Socket的用法,Java中的Socket分为普通Socket和NioSocket两种。

4.1  普通Socket用法

        Java中的网络通信通过Socket实现的,Socket分为ServerSocket和Socket两大类。ServerSocket用于服务端,可以通过accept()方法监听请求返回Socket,Socket用于具体完成数据传输。客户端直接使用Socket发起请求并传输数据。

        ServerSocket的使用分为三步:

        1)创建ServerSocket。

        2)调用创建出来的ServerSocket的accept方法。

        3)使用accept方法返回Socket与客户端进行通信。

        Socket客户端使用:创建一个Socket,和服务器进行通行,传输数据,释放连接。

4.2  NioSocket的用法

        ServerSocketChannel和SocketCHannel,分别对应以前的ServerSocket和Socket。

        NioSocket中服务端的处理过程步骤:

        1)创建ServerSocketChannel并设置相应采参数

        2)创建Selector并注册到ServerSocketChannel

        3)调用Selector的select方法等待请求

        4)Selector接收请求后使用selectedKeys返回SelectionKey集合

        5)使用SelectionKey获取到Channel、Selector和操作类型并进行具体的操作。



第5章  自己动手实现HTTP协议

第6章  详解Servlet

        Servlet是Server+Applet的缩写,表示一个服务器应用。Servlet是一套规范。按照此规范编写的代码就能在java的服务器上运行。

Servlet结构图

6.1  Servlet接口

Servlet接口定义

        init方法:容器启动时被容器调用,只会调用一次。

        getServletConfig方法:用于获得ServletConfig.

        service方法:用于具体处理一次请求。

        getServletInfo方法:获取一些Servlet相关信息,如作者、版权等。

        destroy方法:主要用于Servlet销毁时释放一些资源,只会调用一次。

        在web.xml中定义Servlet时通过init-param标签配置的参数就是通过Servlet来保存的。Tomcat中的Servletinit方法是在ocrg.apache.catalina.core.StandardWrapper的initServlet方法中调用的,ServletConfig传入的是StandardWrapper自身的门面类StandardWrapperFacade。

ServletConfig接口

        ServletConfig(Servlet级别)和ServletContext(Application级别)是最常见的传递初始化参数的。

6.2  GenericServlet

        GenericServlet是Servlet的默认实现,主要做了三件事:实现ServletConfig接口,可以直接调用ServletConfig里面的方法;提供了无参的init方法;提供了log方法。

        GenericServlet实现了ServletConfig,可以在需要用ServletConfig中的方法的时候可以直接调用,而不需要先获取ServletConfig,不过底层还是使用ServletConfig进行调用。

         GenericServlet实现了Servlet的init(ServletConfig  config)方法,现将config设置给了内部变量config,然后调用了无参的init()方法,这个方法是模板方法,子类可以通过覆盖它来完成自己的初始化工作。

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

6.3  HttpServlet

        HttpServlet是HTTP协议实现的Servlet的基类,HttpServlet主要重写了service方法,首先将ServletRequest和ServletResponse转换为了HttpServletRequest和HttpServletResponse,然后根据http请求的类型不同将请求路由到了不同的处理方法。



第7章 Tomcat分析

7.1  Tomcat的顶层结构及启动过程

7.1.1  Tomcat的顶层结构

        Tomcat最顶层的容器时Server,代表整个服务器,Server中包含至少一个Service,应用于具体提供服务。Service主要包含:Connector和Container。Connector用于处理连接相关的事情,并提供Socket与request、response的转换;Container用于封装和管理Servlet以及具体的request请求。一个Tomcat只有一个Server,一个Server可以包含多个Service,一个Service包含多个Connector,但只包含一个Container。

Tomcat整体结构图

        Tomcat里的Server有Catalina来管理,里面有三个方法:load、start、stop,分别用来管理整个服务器的生命周期。Tomcat的入口main方法在Bootstrap中。

7.1.2  Bootstrap的启动过程

        Bootstrap是Tomcat的入口,正常情况下Tomcat就是调用Bootstrap的main方法:

        main方法主要有两部分内容:首先新建了Bootstrap,并执行init方法初始化;然后处理main方法传入的命令,如果args参数为空,默认执行start方法。

        在init方法中初始化了ClassLoader,并使用ClassLoader创建Catalina实例,然后赋值给catalinaDaemon变量。对start的处理,通过反射调用了Catalina的setAwait(true)、load(args)、start()三个方法。

7.1.3  Catalina的启动过程

        从前面可知,catalina的启动主要是调用setAwait、load、start来完成的。setAwait用于设置Server启动完成之后是否进入等待状态标志;load用于加载配置文件,创建并初始化Server;start用于启动服务器。

setAwait方法
load方法
start方法

7.1.4  Server的启动过程

        Server接口中提供addService(Service service)、removeService(Service service)来添加和删除Service,Server的init和start方法分别循环调用每个Service的init和start方法来启动所有的Service。

        Server的默认实现是StandardServer,StandardServer继承自LifecycleMBeanBase,LifecycleMBeanBasej继承自LifecycleBase,init和start方法就在LifecycleBase中,LifecycleBase里的init和start又调用了initInternal和startInternal方法,这两个方法是模板方法。所以调用StandardServer的init和start方法会执行StandardServer自己的initInternal和startInternal方法,这就是Tomcat生命周期的管理方式。

        StandardServer还实现了await方法,Catalina中就是让调用它的服务器进入等待执行状态。

7.1.5  Service的启动过程

        Service默认实现是:StandardService,继承自LifecycleMBeanBasej,init和start方法最终会调用initInternal和startInternal方法.

        现在整个Tomcat服务器就启动了,整个启动流程:

Tomcat顶层组件启动时序图

7.2  Tomcat的生命周期管理

7.2.1  Lifecycle接口

        Tomcat通过org.apache.catalina.Lifecycle接口统一管理生命周期,所有有生命周期的组件都需要实现Lifecycle接口。

Lifecycle接口定义

7.2.2  LifecycleBase

        LifecycleBase默认实现是org.apache.catalina.util.LifecycleBase,所有实现了生命周期的组件都直接或者间接的继承LifecycleBase。LifecycleBase为Lifecycle里的接口方法提供了默认实现:监听器管理使用LifecycleSupport类完成。

        三个管理监听器的方法:

        四个生命周期方法:


        两个获取当前状态的方法:

7.3  Container

7.4  Pipeline-Value管道

7.5  Connector分析



第二篇  俯视Spring MVC

第8章 Spring MVC之初体验

        本章主要介绍SpringMVC的环境搭建并简单介绍使用方法。

8.1  环境搭建

8.2  Spring MVC最简单的配置

        配置一共需要三步:在web.xml中配置Servlet;创建SpringMVC的xml文件;创建Controller和view。

8.2.1  在web.xml中配置Servlet


8.2.2  创建SpringMVC的xml文件

8.2.3  创建Controller和view

8.3  关联sprin源代码



第9章 创建SpringMVC之器

        本章分析SpringMVC创建过程

9.1  整体结构介绍

        Spring MVC中核心Servlet的继承结构:

Spring MVC中核心Servlet的继承结构

9.2  HttpServletBean

        通过前面对Servlet的分析,可知道Servlet创建可以直接调用无参的init方法,HttpServletBean的init方法如下:

        在HttpServletBean的init中,首先,将Servlet中配置的参数使用BeanWrapper设置到DispatcherServlet的相关属性;然后调用模板方法initServletBean,子类通过该方法初始化。

9.3  FrameworkServlet

        从HttpServletBean可知,FrameworkServlet的入口是initServletBean,

        FrameworkServlet主要做了两件事:初始化WebApplicationContext;初始化FrameworkServlet,initFrameworkServlet是个模板方法,子类可以覆盖后在里面做一些初始化的工作,但是子类并没有使用它。

        initWebApplicationContext方法做了三件事:1)获取spring的根容器rootContext;    2)设置WebApplicationContext并根据情况调用onRefresh方法;    3)将WebApplicationContext设置到ServletContext中

        获取spring的根容器rootContext

        默认情况下spring会将自己的容器设置成ServletContext的属性,默认的根容器的key是org.springframwork.web.context.WebApplicationContext.ROOT,它的定义在org.springframwork.web.context.WebApplicationContext中。

       设置WebApplicationContext并根据情况调用onRefresh方法

        设置WebApplicationContext方法有三:

            (1) 在构造方法中已经传递WebApplicationContext参数,这是只需要进行一些设置就可以了。

            (2)WebApplicationContext已经存在在ServletContext中,这时只需要配置Servlet时将ServletContext中的WebApplicationContext的name配置到contextAttribute属性就可以了。例如ServletContext中有个叫haha的WebApplicationContext,可以如下设置:

            (3)自己创建一个WebApplicationContext。正常情况下使用的该方式。创建过程在createWebApplicationContext方法中,createWebApplicationContext内部有调用了configureAndRefreshWebApplicationContext方法:

        需要注意的是,在configureAndRefreshWebApplicationContext方法中为wac添加了监听器,

        SourceFilteringListener可以根据参数进行选择,所以实际监听的是ContextRefreshListener所监听的事件。ContextRefreshListener是FramewoServlet的内部类,监听ContextRefreshedEvent事件,

        再回到initWebApplicationContext方法,可以看到后面根据refreshEvenReceived标志来判断是否要运行onRefresh.

        设置WebApplicationContext的三种方法中,第一种和第三种都已经refresh过了,只有第二种没有,会调用onRefresh。onRefresh只会别调用一次,而DispatcherServlet正是通过重写该方法实现初始化的。

        将WebApplicationContext设置到ServletContext中

        最后,根据publishContext标志判断是否将创建出来的WebApplicationContext设置到ServletContext中。

9.4  DispatcherServlet

        onRefresh方法是DispatcherServlet的入口方法。

        initStrategies就是初始化9个组件,以LocalResolver为例分析具体的初始化方式:

需要注意的是,context指的是WebApplicationContext,而不是ServletContext。

        className来自classNames,calssNames又来自value,value来自defaultStragies.getProperty(key).

        defaultStragies是DispatcherServlet类所在包下的DEFAULT_STRATEGES_PATH文件定义的属性,值是DispatcherServlet.properties。

9.5  小结

        HttpServletBean:作用是将Servlet中配置的参数设置到相应的属性

        FreamworkServlet:初始化WebApplicationContext、初始化FreamworkServlet

        DispatcherServlet:初始化自身的9个组件



第10章  SpringMVC之用

        本章分析SpringMVC是如何运行的,是怎么处理请求的。首先分析HttpServletBean、FreamworkServlet和DispatcherServlet这三个Servlet,目的是明白Servlet容器将请求交

10.1  HttpServletBean

        HttpServletBean在处理请求中没有涉及相应的请求处理。

10.2  FrameworkServlet

        Servlet处理过程:从Servlet的service方法开始,然后再HttpServlet的service方法中根据不同类型将请求路由到doGet、doHead、doPost、doPut、doDelete、doOption、doTrace,并做了doHead、doOption、doTrace的默认实现。

        FrameworkServlet重写了service、doGet、doPost、doPut、doDelete、doOption、doTrace,在service增加了对PATCH类型的处理

        以下是service和doGet的代码:

        由上可以看出,将请求合并到processRequest中。

        LocaleContext:存放着Locale(本地化信息);RequestAttributes:是spring的一个接口,通过它可以get/set/removeAttribute,根据scope参数判断操作request还是session,具体使用的是ServletRequestAttributes,ServletRequestAttributes里面封装了request、response、session,通过get就可以获得。

        LocaleContextHolder,是抽象类,里面的方法是static的,可以直接使用。

        RequestContextHolder也是一样的道理。

        FrameworkServlet工作流程:首先,在service方法中添加对PATCH的处理,并将所需要自己处理的请求集中到processRequest进行统一处理。然后,processRequest将处理逻辑交给模板方法doService,在doService的的前后使用request获取LocaleContexthe和RequestAttributes进行保存,处理完之后恢复。最后,发布ServletRequesthandledEvent事件。

10.3  DispatcherServlet

        DispatcherServlet里面执行处理的入口方法是doService。doService将处理交给doDispatch进行具体处理,在doDispatch处理前doService做了一些事情:首先判断是不是include请求,如果是则对request的Attribute做个快照备份,等doDispatch处理完之后进行还原,

        对request设置的属性中,前面4个属性webApplicationContext、localeResolver、themeResolver和themeSource在Handler和view中使用。后面3个属性都和flashMap相关,主要用于Redirect转发是参数的传递。

        doDispatch的任务:(1)根据request找到Handler    (2)根据Handler找到对应的HandlerAdapter    (3)用HandlerAdapter处理Handler   (4)调用processDispatchResult方法处理上面处理之后的返回结果。

        Hanlder :处理器,直接对应MVC中的C.

        HandlerMapping :用来查找Handler.

        handlerAdapter :适配器,调用具体的Handler对请求进行处理。

10.4  doDispatch结构

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

推荐阅读更多精彩内容