說起spring容器和SpringMVC容器,很多剛接觸spring框架的同志都有點懵,甚至是一頭霧水,分不清楚兩者的關(guān)系和區(qū)別。這倆容器呢雖然有必然的聯(lián)系,但是他們的區(qū)別也是有的。下面我就簡單描述下。
一、前言
首先在我們開發(fā)中會與到各種各樣的容器,今天我們就說一下spring 容器與springmvc容器。Spring和SpringMVC作為Bean管理容器和MVC層的默認(rèn)框架,已被眾多web應(yīng)用采用。但是在實際應(yīng)用中,初級開發(fā)者常常會因?qū)pring和SpringMVC的配置失當(dāng)導(dǎo)致一些奇怪的異?,F(xiàn)象,比如Controller的方法無法攔截、Bean被多次加載等問題,這種情況發(fā)生的根本原因在于開發(fā)者對Spring容器和SpringMVC容器之間的關(guān)系了解不夠深入。
在Spring整體框架的核心概念中,容器的核心思想是管理Bean的整個生命周期。但在一個項目中,Spring容器往往不止一個,最常見的場景就是在一個項目中引入Spring和SpringMVC這兩個框架,其本質(zhì)就是兩個容器。首先 springmvc和spring它倆都是容器,容器就是管理對象的地方,例如Tomcat,就是管理servlet對象的,而springMVC容器和spring容器,就是管理bean對象的地方,再說的直白點,springmvc就是管理controller對象的容器,spring就是管理service和dao的容器,這下你明白了吧。所以我們在springmvc的配置文件里配置的掃描路徑就是controller的路徑,而spring的配置文件里自然配的就是service和dao的路徑
SpringMVC.xml文件所配置的路徑:
applicationContext-service.xml文件所配置的路徑:
其次, spring容器和springmvc容器的關(guān)系是父子容器的關(guān)系。spring容器是父容器,springmvc是子容器。在子容器里可以訪問父容器里的對象,但是在父容器里不可以訪問子容器的對象,說的通俗點就是,在controller里可以訪問service對象,但是在service里不可以訪問controller對象 所以這么看的話,所有的bean,都是被spring或者springmvc容器管理的,他們可以直接注入。然后springMVC的攔截器也是springmvc容器管理的,所以在springmvc的攔截器里,可以直接注入bean對象。
二、Spring容器、SpringMVC容器與ServletContext之間的關(guān)系
在Web容器中配置Spring時,你可能已經(jīng)司空見慣于web.xml文件中的以下配置代碼,下面我們以該代碼片段為基礎(chǔ)來了解Spring容器、SpringMVC容器與ServletContext之間的關(guān)系。要想理解這三者的關(guān)系,需要先熟悉Spring是怎樣在web容器中啟動起來的。Spring的啟動過程其實就是其Spring IOC容器的啟動過程。特別地,對于web程序而言,IOC容器啟動過程即是建立上下文的過程。
…org.springframework.web.context.ContextLoaderListenercontextConfigLocationclasspath:applicationContext.xmlSpringMVCorg.springframework.web.servlet.DispatcherServletcontextConfigLocationclasspath:springmvc.xml1SpringMVC/ …
Spring的啟動過程
Spring容器與SpringMVC的容器聯(lián)系與區(qū)別
ContextLoaderListener中創(chuàng)建Spring容器主要用于整個Web應(yīng)用程序需要共享的一些組件,比如DAO、數(shù)據(jù)庫的ConnectionFactory等;而由DispatcherServlet創(chuàng)建的SpringMVC的容器主要用于和該Servlet相關(guān)的一些組件,比如Controller、ViewResovler等。它們之間的關(guān)系如下:
作用范圍
子容器(SpringMVC容器)可以訪問父容器(Spring容器)的Bean,父容器(Spring容器)不能訪問子容器(SpringMVC容器)的Bean。也就是說,當(dāng)在SpringMVC容器中g(shù)etBean時,如果在自己的容器中找不到對應(yīng)的bean,則會去父容器中去找,這也解釋了為什么由SpringMVC容器創(chuàng)建的Controller可以獲取到Spring容器創(chuàng)建的Service組件的原因。
具體實現(xiàn)
在Spring的具體實現(xiàn)上,子容器和父容器都是通過ServletContext的setAttribute方法放到ServletContext中的。但是,ContextLoaderListener會先于DispatcherServlet創(chuàng)建ApplicationContext,DispatcherServlet在創(chuàng)建ApplicationContext時會先找到由ContextLoaderListener所創(chuàng)建的ApplicationContext,再將后者的ApplicationContext作為參數(shù)傳給DispatcherServlet的ApplicationContext的setParent()方法。也就是說,子容器的創(chuàng)建依賴于父容器的創(chuàng)建,父容器先于子容器創(chuàng)建。在Spring源代碼中,你可以在FrameServlet.java中找到如下代碼:
wac.setParent(parent);
其中,wac即為由DisptcherServlet創(chuàng)建的ApplicationContext,而parent則為有ContextLoaderListener創(chuàng)建的ApplicationContext。此后,框架又會調(diào)用ServletContext的setAttribute()方法將wac加入到ServletContext中。
三、Spring容器和SpringMVC容器的配置
在Spring整體框架的核心概念中,容器是核心思想,就是用來管理Bean的整個生命周期的,而在一個項目中,容器不一定只有一個,Spring中可以包括多個容器,而且容器間有上下層關(guān)系,目前最常見的一種場景就是在一個項目中引入Spring和SpringMVC這兩個框架,其實就是兩個容器:Spring是根容器,SpringMVC是其子容器。在上文中,我們提到,SpringMVC容器可以訪問Spring容器中的Bean,Spring容器不能訪問SpringMVC容器的Bean。但是,若開發(fā)者對Spring容器和SpringMVC容器之間的關(guān)系了解不夠深入,常常會因配置失當(dāng)而導(dǎo)致同時配置Spring和SpringMVC時出現(xiàn)一些奇怪的異常,比如Controller的方法無法攔截、Bean被多次加載等問題。
在實際工程中,一個項目中會包括很多配置,根據(jù)不同的業(yè)務(wù)模塊來劃分,我們一般思路是各負(fù)其責(zé),明確邊界,即:Spring根容器負(fù)責(zé)所有其他非controller的Bean的注冊,而SpringMVC只負(fù)責(zé)controller相關(guān)的Bean的注冊,下面我們演示這種配置方案。
(1). Spring容器配置
Spring根容器負(fù)責(zé)所有其他非controller的Bean的注冊:
(2). SpringMVC容器配置
SpringMVC只負(fù)責(zé)controller相關(guān)的Bean的注冊,其中@ControllerAdvice用于對控制器進(jìn)行增強,常用于實現(xiàn)全局的異常處理類:
在中可以添加use-default-filters,Spring配置中的use-default-filters用來指示是否自動掃描帶有@Component、@Repository、@Service和@Controller的類。默認(rèn)為true,即默認(rèn)掃描。如果想要過濾其中這四個注解中的一個,比如@Repository,可以添加子標(biāo)簽,如下:
而子標(biāo)簽是用來添加掃描注解的:
四、Spring容器和SpringMVC容器的配置失當(dāng)帶來的問題
問題描述
在一個項目中,想使用Spring AOP在Controller中切入一些邏輯,但發(fā)現(xiàn)不能切入到Controller的中,但可以切入到Service中。最初的配置情形如下:
1). Spring的配置文件application.xml包含了開啟AOP自動代理、Service掃描配置以及Aspect的自動掃描配置,如下所示:
2). Spring MVC的配置文件spring-mvc.xml主要內(nèi)容是Controller層的自動掃描配置。
3). 增強代碼為如下:
@Component@Aspectpublic class SecurityAspect {private static final String DEFAULT_TOKEN_NAME = “X-Token”;private TokenManager tokenManager;@Resource(name = “tokenManager”)public void setTokenManager(TokenManager tokenManager) {this.tokenManager = tokenManager;}@Around(“@annotation(org.springframework.web.bind.annotation.RequestMapping)”)public Object execute(ProceedingJoinPoint pjp) throws Throwable {// 從切點上獲取目標(biāo)方法MethodSignature methodSignature = (MethodSignature) pjp.getSignature();Method method = methodSignature.getMethod();// 若目標(biāo)方法忽略了安全性檢查,則直接調(diào)用目標(biāo)方法if (method.isAnnotationPresent(IgnoreSecurity.class)) {System.out.println(“method.isAnnotationPresent(IgnoreSecurity.class) : “+ method.isAnnotationPresent(IgnoreSecurity.class));return pjp.proceed();}// 從 request header 中獲取當(dāng)前 tokenString token = WebContext.getRequest().getHeader(DEFAULT_TOKEN_NAME);// 檢查 token 有效性if (!tokenManager.checkToken(token)) {String message = String.format(“token [%s] is invalid”, token);throw new TokenException(message);}// 調(diào)用目標(biāo)方法return pjp.proceed();}}
4). 需要被代理的Controller如下:
@RestController@RequestMapping(“/tokens”)public class TokenController {private UserService userService;private TokenManager tokenManager;public UserService getUserService() {return userService;}@Resource(name = “userService”)public void setUserService(UserService userService) {this.userService = userService;}public TokenManager getTokenManager() {return tokenManager;}@Resource(name = “tokenManager”)public void setTokenManager(TokenManager tokenManager) {this.tokenManager = tokenManager;}@RequestMapping(method = RequestMethod.POST)@IgnoreSecuritypublic Response login(@RequestParam(“uname”) String uname,@RequestParam(“passwd”) String passwd) {boolean flag = userService.login(uname, passwd);if (flag) {String token = tokenManager.createToken(uname);System.out.println(“**** Token **** : ” + token);return new Response().success(“Login Success…”);}return new Response().failure(“Login Failure…”);}@RequestMapping(method = RequestMethod.DELETE)@IgnoreSecuritypublic Response logout(@RequestParam(“uname”) String uname) {tokenManager.deleteToken(uname);return new Response().success(“Logout Success…”);}}
在運行過程中,發(fā)現(xiàn)這樣配置并沒有起作用,AOP配置不生效,沒有生成TokenController的代理。
解決方案
由上一節(jié)可知,原因有兩點:
- Spring容器與SpringMVC容器分別基于各自的配置文件進(jìn)行初始化,所以,在SpringMVC容器創(chuàng)建TokenController時,由于其沒有啟用AOP代理,導(dǎo)致SpringMVC容器沒有為TokenController生成代理,所以沒有生效。
- 雖然父容器啟用了AOP代理,但由于父子容器的獨立性,無濟于事。
因此,我們只需要在SpringMVC的配置文件中添加Aspect的自動掃描配置即可實現(xiàn)所要的效果。此外,一般地,SpringMVC容器只管理Controller,剩下的Service、Repository 和 Component 由Spring容器只管理,不建議兩個容器上在管理Bean上發(fā)生交叉。因此,建議配置為:
SpringMVC 配置:
Spring配置:
總結(jié)
springmvc容器是spring容器的子容器,但是子容器可以父容器的bean, 但是父容器不能訪問子容器的bean。Spring容器和SpringMVC容器雖然是父容器與子容器的關(guān)系,但二者之間具有一定的獨立性。具體來說,兩個容器基于各自的配置文件分別進(jìn)行初始化,只有在子容器找不到對應(yīng)的Bean時,才回去父容器中去找并加載
寫在最后
希望通過這篇文章能讓大家分清楚spring容器和SpringMVC容器的關(guān)系與區(qū)別。雖然這些知識點都是老生常談,不否認(rèn)還是很多人分不清和不了解,但知識點不管是不是老生常談也都是需要掌握的,畢竟基礎(chǔ)要打好,這樣才能有更高的成就賺到更多的錢呀。好了本文到此結(jié)束了,希望能對鐵子們有幫助和收獲。喜歡的鐵子們可以點點贊和關(guān)注, 文章持續(xù)更新,也可以評論出你想看哪一塊技術(shù)。鐵子們的支持是我的動力,創(chuàng)作離不開鐵子們的支持,在此先感謝大家!