[090]web容器启动探讨
tomcat 如何启动spring容器
我们知道spring通过容器来管理bean,在spring容器启动的时候会实例化所有的bean包括从xml读取或者annotation扫描得到,即所有的对象live in applicationContext.
我们想象这样一个场景,启动web容器如何启动spring applicatin Context,它的entrance在哪?
tomcat中,我们通过listener来启动spring 容器。
以下是我们工程中常用的web.xml结构
web.xml (参考fs-sail-provider/web.xml)
<context-param>
<!--如果applicationContext在src/main/webapp/WEB-INF下该参数可以不用写-->
<param-name>contextConfigLocation</param-name>
<param-value>classpath:spring/applicationContext.xml</param-value>
</context-param>
<listener>
<listener-class>org.springframework.web.context.ContextLoaderListener</listener-class>
</listener>
<servlet>
<servlet-name>springmvc</servlet-name>
<servlet-class>org.springframework.web.servlet.DispatcherServlet</servlet-class>
<load-on-startup>1</load-on-startup>
</servlet>
<servlet-mapping>
<servlet-name>springmvc</servlet-name>
<url-pattern>/*</url-pattern>
</servlet-mapping>
可以看到Spring容器的entrance触发是通过ContextLoaderListener来触发的,Spring Mvc的入口是DispatcherServlet。
至于DispatcherServlet的实现机制我们可以看 一下它的类族图(intellij中通过ctrl+h调出类图)。
image.png
如何启动resteasy-jaxrs
<listener>
<listener-class>
org.jboss.resteasy.plugins.server.servlet.ResteasyBootstrap
</listener-class>
</listener>
<listener>
<listener-class>
com.facishare.paas.appframework.fcp.model.FcpRestServerContextListener
</listener-class>
</listener>
<servlet>
<servlet-name>Resteasy</servlet-name>
<servlet-class>
org.jboss.resteasy.plugins.server.servlet.HttpServletDispatcher
</servlet-class>
</servlet>
<servlet-mapping>
<servlet-name>Resteasy</servlet-name>
<url-pattern>/API/*</url-pattern>
</servlet-mapping>
与spring中的dispatcher servlet一样,jboss的httpServletDispatcher也是继承了javax.servlet.http.HttpServlet
image.png
由于该项目也是 基于spring做了一些定制开发,需要启动spring容器,其自定义的FcpRestServerContextListener的类族图如下:
image.png
如何加载引用jar包中的bean
如何工程A引用了工程B的jar包,如何让工程B的spring bean注入进去呢。
由于spring容器启动的入口只有一个就是Listener对应的ContextLoaderListener,其对应的配置文件为applicationContext.xml。
所以我们必须在A的applicationContext.xml钟加上 B的配置文件。
类似如:
....
<import resource="classpath:spring/metadata.xml"/>
<import resource="classpath:spring/common.xml"/>
对于dubbo的bean的引用形式怎么样的呢?
在谈到远程引用的情况,我们先谈谈本地引用的情况:
在本地spring容器中,对于如下类
@Service
Class Bean2
{
@autuowiere
Bean1 bean1;
}
在启动spring容器中会做两件事,先通过spring容器 instant Bean2,然后instant Bean2发现Bean2包含有bean1的成员且需要注入(注入就是变量与实例关联的过程),那么就去spring容器中找bean1的实例如果找不到就会报错。所以这里其实包含两个过程a).bean的实例化,b).bean的注入过程。
那么对应dubbo远程服务同样包含,客户端与服务端bean的实例化和关联的过程。下面我们先讲服务端bean的实例化,然后再讲客户端如何引用它。
首先对于工程A调用工程B对应的服务,工程A只引用工程B对应的api jar包,通过A引用B定义的服务达到远程访问的目的。
1).服务端:在B的applicationContext中必须定义该服务的实现
<dubbo:service interface="com.facishare.sail.api.service.AuthService" ref="authService" protocol="dubbo"/>
<!--authService 为B工程中实例化的bean-->
其中,B工程中 authService bean定义如下:
@Service("authService")
public class AuthServiceImpl implements AuthService {
......
}
所以我们知道,dubbo:service是把spring容器中的bean authService注册到注册中心供远程调用。
2 ).客户端:在A中的applicatonContext引用B的服务:
<dubbo:reference interface="com.facishare.sail.api.service.AuthService" id="authService"/>
引用了B的服务后,其A工程如何使用呢?请参考如下代码:
public class AuthController {
@Autowired
private AuthService authService; //通过id="authService"去注册中心查找服务
}
说在后面的话
对于Spring dispatcherServlet这个入口如何启作用,为什么继承了httpServlet就会使用dispatcherServlet估计需要了解下tomcat底层的源代码了。
学习一个东西把逻辑想清楚,不要貌似的以为懂了,在学习上需要闭环。