生活随笔
收集整理的這篇文章主要介紹了
spring里头各种获取ApplicationContext的方法
小編覺得挺不錯的,現在分享給大家,幫大家做個參考.
為啥寫這個文章呢?spring各個版本不同,以及和系統框架套在一起不同,導致獲取的方式不同,網絡上各種版本,太亂了,寫獲取方式的人都不寫這個獲取方式是在本地還是在WEB,在那種應用服務器下,在spring那個版本下,太過分了!
我這寫一些,常見的,可能經常要用的版本;
首先了解,為什么要獲取這個東西 :當你想通過spring獲取一個你指定的類的實例的時候,而又沒有通過spring加載到當前調用的類里面,例如你在filter里面,可能要對人員角色做判定,此時還沒到業務層代碼,但是又要訪問數據庫或其他的服務類。
然后再確保一點 :這個context是一個全局變量,spring加載的時候,根handle信息就被裝載,無論是本地應用程序還是web應用都是這樣,下面分別說下如果是本地程序和其他情況的獲取方式。
如果是main方法,你要啟動spring,有很多方法,有基于annotation的注解來講配置文件裝載起來,當然,你想獲取applicationCntext可在main方法中這樣獲取:
[java] ?view plaincopy
XmlBeanFactory?factory?=? new ?XmlBeanFactory( new ?ClassPathResource( "beans.xml" )); ??
還有沒有其他的方式呢?有的
[java] view plaincopy
ApplicationContext?context?=? new ?ClassPathXmlApplicationContext( new ?String[]?{ "a.xml" ,? "b.xml" });??
還有沒有其他的?有
[java] view plaincopy
XmlWebApplicationContext?context?=? new ?XmlWebApplicationContext();?? context.setConfigLocations(new ?String[]?{ "aaa.xml" ?,? "bb.xml" });?? MockServletContext?msc?=?new ?MockServletContext();?? context.setServletContext(msc);?? context.refresh();?? msc.setAttribute(WebApplicationContext.ROOT_WEB_APPLICATION_CONTEXT_ATTRIBUTE,?context);??
其實方法差不多,他們有著繼承關系,所以方法很多,你每次new的時候,相當于重新創建一個
applicationContext ,他會重新裝載,所以不適合反復調用,如果自己new,你就應當把它放到一個全局變量中,用main啟動的,當然你通過直接或間接的
static 應用到這個
application 即可。
而在WEB上呢,有一種是通過spring來加載spring本身的方式是:
通過實現接口:
[java] view plaincopy
org.springframework.context.ApplicationContextAware??
然后spring反射,來源文章:http://blog.163.com/xuyang1974@126/blog/static/2684016320101028101923914/
這種方式適在spring 2、3當中均有效:
編寫類:
[java] view plaincopy
import ?org.springframework.beans.BeansException;?? import ?org.springframework.context.ApplicationContext;?? import ?org.springframework.context.ApplicationContextAware;?? import ?org.springframework.stereotype.Service;?? @Service ?? public ? class ?SpringContextHolder? implements ?ApplicationContextAware?{?? ????private ? static ?ApplicationContext?applicationContext;?? ?? ????@Override ?? ????public ? void ?setApplicationContext(ApplicationContext?applicationContext)? throws ?BeansException?{?? ????????SpringContextHolder.applicationContext?=?applicationContext;?? ????}?? ?? ?? ????public ? static ?ApplicationContext?getApplicationContext()?{?? ????????return ?applicationContext;?? ????}?? ????public ? static ?Object?getBean(String?beanName)?{?? ????????return ?applicationContext.getBean(beanName);?? ????}?? ?????? ????public ? static ?<T>T?getBean(String?beanName?,?Class<T>clazz)?{?? ????????return ?applicationContext.getBean(beanName?,?clazz);?? ????}?? }??
我這里是通過annotation注解的,如果不是annotation,那么可以通過配置文件:
[html] view plaincopy
< bean ? class = "xxx.xxx.xxx.SpringContextHolder" > </ bean > ??
來進行注入操作,結果一樣,如果的spring配置中,沒有設置byName 的話,bean的配置里面記得要加參數來設置applicationContext來反射進去。
而你要加載spring,很多時候,并不是進入業務層的,因為反射是反射到業務層的,你還沒有進入業務層,怎么來獲取這個反射的東西呢?除非你反射的時候,用static變量來獲取,那么就沒有問題了;所以上面的例子中他也用的是static;
當你不想用static來反射,而經常想要用到它的時候,就有很多種獲取方式了。
在spring 3以前 的版本,我們在WEB應用中通常是這樣獲取的:
[java] view plaincopy
WebApplicationContext?wac?=?WebApplicationContextUtils.getWebApplicationContext(context);??
而contexnt是什么呢?如果是servlet中,是可以直接通過getServletContext()獲取,
而通過request要這樣獲取:
對于所有的tomcat通用的寫法是:
[java] view plaincopy
ServletContext?context?=?req.getSession().getServletContext();??
對于
tomcat 7 以上的寫法是(也就是tomcat 7可以直接從request中獲取servletContext,tomcat6不行,必須通過session才可以):
[java] view plaincopy
ServletContext?context?=?req.getServletContext();??
其實從spring 3過后,獲取的方法就有所改變,變得很詭異,因為竟然不兼容以前的獲取方法,spring 3當中將其進行了進一步的包裝,你在其他地方可能看到各種各樣的版本。
spring 2中之所以可以那樣獲取,是因為spring 2當中通常會配置一個listener,由他來加載spring,他在filter之前;spring 3當中,通過org.springframework.web.servlet.DispatcherServlet 來裝載spring的信息,初始化在其父親類:org.springframework.web.servlet.FrameworkServlet中方法:initWebApplicationContext();
跟蹤方法明顯看到內部獲取增加了一個參數:
[java] ?view plaincopy
WebApplicationContext?wac?=?WebApplicationContextUtils.getWebApplicationContext(getServletContext(),attrName);??
這個參數是什么呢?
經過跟蹤可以發現是:
[java] ?view plaincopy
FrameworkServlet.SERVLET_CONTEXT_PREFIX?+?getServletName()??
而SERVLET_CONTEXT_PREFIX的定義是:
[java] ?view plaincopy
public ? static ? final ?String?SERVLET_CONTEXT_PREFIX?=?FrameworkServlet. class .getName()?+? ".CONTEXT." ;??
也就是:
[java] ?view plaincopy
“org.springframework.web.servlet.FrameworkServlet.CONTEXT.”??
而getServletName()呢?他是當前請求的servlet,可以獲取到的一個web.xml里面配置的名稱,例如,
如果你的web.xml中配置的是:
[html] ?view plaincopy
< servlet > ?? ????????< servlet-name > spring </ servlet-name > ?? ????????< servlet-class > org.springframework.web.servlet.DispatcherServlet </ servlet-class > ?? ????????< load-on-startup > 1 </ load-on-startup > ?? ????</ servlet > ??
說明getServletName()的結果就是spring,否則就是其他,那么如果是spring,就是:
[java] ?view plaincopy
org.springframework.web.servlet.FrameworkServlet.CONTEXT.spring??
ok,如果按照上面的配置,獲取方式就是:
[java] ?view plaincopy
request.getSession().getServletContext().getAttribute( "org.springframework.web.servlet.FrameworkServlet.CONTEXT.spring" );??
tomcat 7以上可以寫成:
[java] ?view plaincopy
request.getServletContext().getAttribute( "org.springframework.web.servlet.FrameworkServlet.CONTEXT.spring" );??
更為好的寫法是:
[html] ?view plaincopy
request.getSession().getServletContext().getAttribute(FrameworkServlet.SERVLET_CONTEXT_PREFIX?+"spring");??
以下為spring為了方便,做的一些擴展:
spring為了業務代碼中獲取這個參數方便,在進入業務代碼前做了一個操作,在DispatcherServlet的方法:doService中doDispatch調用之前:
[java] ?view plaincopy
request.setAttribute(WEB_APPLICATION_CONTEXT_ATTRIBUTE,?getWebApplicationContext());??
也就是,當你進入Controller以后,獲取就不用那么麻煩了,你只需要這樣就能獲取到:
[java] ?view plaincopy
request.getAttribute(DispatcherServlet.WEB_APPLICATION_CONTEXT_ATTRIBUTE);??
當然,你可以將值寫進去,看定義是:
[java] ?view plaincopy
public ? static ? final ?String?WEB_APPLICATION_CONTEXT_ATTRIBUTE?=?DispatcherServlet. class .getName()?+? ".CONTEXT" ;??
那么值就應該是:
[java] ?view plaincopy
org.springframework.web.servlet.DispatcherServlet.CONTEXT??
所以在Controller中你還可以這樣來獲取:
[java] ?view plaincopy
request.getAttribute( "org.springframework.web.servlet.DispatcherServlet.CONTEXT" )??
經過spring包裝后,你也可以通過:
[java] ?view plaincopy
RequestContextUtils.getWebApplicationContext(request?,?context)??
來獲取,源碼如下:
其實它獲取的方式和上面給的方法是一樣的,RequestContextUtils.getWebApplicationContext 在spring 3當中,如果沒有啟動ContextLoaderListener (當然你可以配置監聽),是不會成功的。
ContextLoaderListener 的簡單配置為(web.xml中):
[html] ?view plaincopy
< listener > ?? ?< listener-class > org.springframework.web.context.ContextLoaderListener </ listener-class > ?? </ listener > ??
spring 3以后基本不這樣配置了。
from:?http://blog.csdn.net/xieyuooo/article/details/8473503
總結
以上是生活随笔 為你收集整理的spring里头各种获取ApplicationContext的方法 的全部內容,希望文章能夠幫你解決所遇到的問題。
如果覺得生活随笔 網站內容還不錯,歡迎將生活随笔 推薦給好友。