大牛请绕过,此文仅针对自己小白水平,对web程序的启动流程做个清晰的回顾。
一.使用spring等框架的web程序在Tomcat下的启动流程
1)Tomcat是根据web.xml来启动的。首先到web.xml
2)web.xml中负责启动spring和spring mvc。对应的启动配置文件分别是
启动spring mvc,并进行所有资源路径映射
<servlet> <servlet-name>springMVC</servlet-name> <servlet-class>org.springframework.web.servlet.DispatcherServlet</servlet-class> <init-param> <param-name>contextConfigLocation</param-name> <param-value>/WEB-INF/cfg/springmvc-servlet.xml</param-value> </init-param> <load-on-startup>1</load-on-startup> </servlet> <servlet-mapping> <servlet-name>springMVC</servlet-name> <url-pattern>/</url-pattern> </servlet-mapping>
启动spring,通过ContextLoaderListener
<listener> <listener-class>org.springframework.web.context.ContextLoaderListener</listener-class> </listener> <context-param> <param-name>contextConfigLocation</param-name> <param-value>/WEB-INF/cfg/spring.xml</param-value> </context-param>
3)spring中对一些orm框架的启动,包括Mybatis/hibernate。orm框架的启动基本都是通过sqlsessionFactory bean来启动的。
并配置各种bean到ioc容器中。包括datasource等。
4)web应用程序中,spring相当于程序运行的平台,spring对整个程序提高供ioc支持和aop支持。
spring提供注解如@service @repository @component将各种类注册到ioc容器中。通过设置scan package的方式,spring在启动时候会扫描包下的所有注解,并将它们注册到ioc容器中。并针对@autowired @resource,将一些bean从ioc容器中获取填充到bean的构造属性中。
spring会自动扫描如下包中的注解:
<context:component-scan base-package=”pagkage1[,pagkage2,…,pagkageN]”/>
以上可以看出spring所有的bean注入都是在spring.xml中配置的,所有的bean注入都在spring.xml中配置的。
@autowired @resource只针对于类的成员变量,不针对方法里的局部变量。
注:正是spring的ioc支持了controller层注入service,service注入dao。打通了各层之间的桥梁,省去了原来的new service(),new Dao()的方法。
问答
1.使用spring ioc有什么好处?
答:网上有很多介绍使用Spring IOC容器的好处的文章,我看到的主要有两点,
1.方便测试,当需要测试的类依赖其他的类时,可以为依赖的类做个mock,然后注入进来。
2.方便维护及升级,当某个类需要修改实现时,只要接口没变,则无需修改源代码,重写一个实现类,修改下配置即可。
3.默认下,IOC容器管理的bean都是单例的,不会被gc掉。
4.spring提倡面向接口开发,因此在servcie里面的dao直接是接口就可以了。这样有助于解耦。(测试等等)。。
如果不用spring,要么自己new daoimpl,就面向具体了,如果还想抽象,就必须自己写工厂——其实spring就是一个工厂IOC。实现类似的作用。
其实,面向抽象是实现和接口分离,对于多数的项目,实现大量变化的机会不是很大,因此直接面向具体(省略掉dao接口等)是我个人推荐的做法。
spring工厂对其中的对象的生命周期,作用范围都有很明确的定义,很容易使用,如果自己做这么一套,只怕没这么好。
第三,spring对ioc管理的对象,可以做很多增强,比如aop,代理等等。
5.感觉AOP最有用了。
2.spring是怎样支持面向接口编程的?
由于依赖于接口,可以通过依赖注入随时替换DAO接口的实现类,而应用程序完全不用了解接口与底层数据库操作细节。
应用程序----调用-----》DAO接口《---实现--DAO接口实现类----操作------》数据库