Springプロジェクトを起動する詳細プロセス(小結)
1、Springプロジェクトはwebプロジェクト容器に入れる(Tomcat、Jetty、JBoss)
本論文は共通のTomcatを例にとっている。
2、プロジェクト容器の起動時にweb.xmlのプロファイルを読み込む必要があります。
次の図のように
3、容器はまずweb.xmlプロファイルの二つのノードを読みます。 と
説明:
tomcatはウェブコンテナを起動する時にServlet Contact Listenerというモニターを起動します。ウェブコンテナの中にServlet Contact Listenerというインターフェースが実装されるたびに、ウェブコンテナはServlet ContitListenerがインスタンスの対象によってそのcontextInitiazedを実行するように通知します。
springフレームは設計の過程でContect LoadListenerというクラスがServlet Contact Listenerというインターフェースを実現したので、Contect LoadListenerという種類が実用化されるたびに、webコンテナはSpringにcontextInitialized()を実行するように通知して、spring容器の作成過程に進みます。
4、Contect LoaderListenerのcontext Initialized()はspring容器の起動配置を行い、initWebAppliation Contectを呼び出してspring容器を初期化する;
configreAndRefresh WebAppleicationControtextにspringをロードするプロファイル、すなわちweb.xmlにを読み込みます。
または
springプロファイルを以下のコードでロードします。
トップインターフェースResource Loaderは一つのget Resource(String location)方法を提供しており、一つのリソースアドレスに従ってリソースファイルをロードすることができます。リソースアドレスの表現は以下のようになります。
--1.クラスパスパス:プレフィックス先頭の表現、例えば、クラスパスパス:smart-context.xml
--2.「/」の先頭の表現は、例えば、/WEB-INF/clases/smart-context.xml
--3.「/」ではない冒頭の表現、例えば、WEB-INF/clases/smart-context.xml
--4.urlプロトコル、例えば、file:/D:/ALANWANG-AIA/Horse-workspace/chapter 3/target/clases/smart-context.xml
SpringはDefault Resource Loaderを提供しています。Default Resource Loaderは以上列挙した機能を実現した上で、開発者にカスタム拡張インターフェースProtocolverを提供しています。開発者はこのインターフェースのカスタマイズされた個性的なリソース表現を実現できます。コードは以下の通りです。
まずProtocoll Resoloverインターフェースを実現します。
以上が本文の全部です。皆さんの勉強に役に立つように、私たちを応援してください。
本論文は共通のTomcatを例にとっている。
2、プロジェクト容器の起動時にweb.xmlのプロファイルを読み込む必要があります。
次の図のように
3、容器はまずweb.xmlプロファイルの二つのノードを読みます。
説明:
tomcatはウェブコンテナを起動する時にServlet Contact Listenerというモニターを起動します。ウェブコンテナの中にServlet Contact Listenerというインターフェースが実装されるたびに、ウェブコンテナはServlet ContitListenerがインスタンスの対象によってそのcontextInitiazedを実行するように通知します。
springフレームは設計の過程でContect LoadListenerというクラスがServlet Contact Listenerというインターフェースを実現したので、Contect LoadListenerという種類が実用化されるたびに、webコンテナはSpringにcontextInitialized()を実行するように通知して、spring容器の作成過程に進みます。
4、Contect LoaderListenerのcontext Initialized()はspring容器の起動配置を行い、initWebAppliation Contectを呼び出してspring容器を初期化する;
@Override
public void contextInitialized(ServletContextEvent event) {
initWebApplicationContext(event.getServletContext());
}
public WebApplicationContext initWebApplicationContext(ServletContext servletContext) {
//Spring ,spring
if(servletContext.getAttribute(WebApplicationContext.ROOT_WEB_APPLICATION_CONTEXT_ATTRIBUTE) != null) {
throw new IllegalStateException("Cannot initialize context because there is already a root application context present - check whether you have multiple ContextLoader* definitions in your web.xml!");
} else {
Log logger = LogFactory.getLog(ContextLoader.class);
servletContext.log("Initializing Spring root WebApplicationContext");
if(logger.isInfoEnabled()) {
logger.info("Root WebApplicationContext: initialization started");
}
long startTime = System.currentTimeMillis();
try {
// spring ( spring bean)
if(this.context == null) {
this.context = this.createWebApplicationContext(servletContext);
}
if(this.context instanceof ConfigurableWebApplicationContext) {
ConfigurableWebApplicationContext cwac = (ConfigurableWebApplicationContext)this.context;
if(!cwac.isActive()) {
if(cwac.getParent() == null) {
ApplicationContext parent = this.loadParentContext(servletContext);
cwac.setParent(parent);
}
//Spring , spring
this.configureAndRefreshWebApplicationContext(cwac, servletContext);
}
}
servletContext.setAttribute(WebApplicationContext.ROOT_WEB_APPLICATION_CONTEXT_ATTRIBUTE, this.context);
ClassLoader ccl = Thread.currentThread().getContextClassLoader();
if(ccl == ContextLoader.class.getClassLoader()) {
currentContext = this.context;
} else if(ccl != null) {
currentContextPerThread.put(ccl, this.context);
}
if(logger.isDebugEnabled()) {
logger.debug("Published root WebApplicationContext as ServletContext attribute with name [" + WebApplicationContext.ROOT_WEB_APPLICATION_CONTEXT_ATTRIBUTE + "]");
}
if(logger.isInfoEnabled()) {
long elapsedTime = System.currentTimeMillis() - startTime;
logger.info("Root WebApplicationContext: initialization completed in " + elapsedTime + " ms");
}
return this.context;
} catch (RuntimeException var8) {
logger.error("Context initialization failed", var8);
servletContext.setAttribute(WebApplicationContext.ROOT_WEB_APPLICATION_CONTEXT_ATTRIBUTE, var8);
throw var8;
} catch (Error var9) {
logger.error("Context initialization failed", var9);
servletContext.setAttribute(WebApplicationContext.ROOT_WEB_APPLICATION_CONTEXT_ATTRIBUTE, var9);
throw var9;
}
}
}
5、spring容器の作成が完了したら、実用化を開始してbeanをロードし、Spring容器の作成が完了したら、spring容器にbeanをロードしてconfigreAndRefresh WebAppleicationContextを使用するつもりです。beanのロードを完了します。
protected void configureAndRefreshWebApplicationContext(ConfigurableWebApplicationContext wac, ServletContext sc) {
if (ObjectUtils.identityToString(wac).equals(wac.getId())) {
// The application context id is still set to its original default value
// -> assign a more useful id based on available information
String idParam = sc.getInitParameter(CONTEXT_ID_PARAM);
if (idParam != null) {
wac.setId(idParam);
}
else {
// Generate default id...
wac.setId(ConfigurableWebApplicationContext.APPLICATION_CONTEXT_ID_PREFIX +
ObjectUtils.getDisplayString(sc.getContextPath()));
}
}
wac.setServletContext(sc);
String configLocationParam = sc.getInitParameter(CONFIG_LOCATION_PARAM);
if (configLocationParam != null) {
wac.setConfigLocation(configLocationParam);
}
// The wac environment's #initPropertySources will be called in any case when the context
// is refreshed; do it eagerly here to ensure servlet property sources are in place for
// use in any post-processing or initialization that occurs below prior to #refresh
ConfigurableEnvironment env = wac.getEnvironment();
if (env instanceof ConfigurableWebEnvironment) {
((ConfigurableWebEnvironment) env).initPropertySources(sc, null);
}
customizeContext(sc, wac);
wac.refresh();
}
説明:configreAndRefresh WebAppleicationControtextにspringをロードするプロファイル、すなわちweb.xmlに
または
springプロファイルを以下のコードでロードします。
public class Application{
public static void main(String[] args) {
ClassPathXmlApplicationContext ctx = new ClassPathXmlApplicationContext("/context.xml");
ctx.start();
}
}
ここではDefault Resource Loaderを呼び出す方法を略しています。トップインターフェースResource Loaderは一つのget Resource(String location)方法を提供しており、一つのリソースアドレスに従ってリソースファイルをロードすることができます。リソースアドレスの表現は以下のようになります。
--1.クラスパスパス:プレフィックス先頭の表現、例えば、クラスパスパス:smart-context.xml
--2.「/」の先頭の表現は、例えば、/WEB-INF/clases/smart-context.xml
--3.「/」ではない冒頭の表現、例えば、WEB-INF/clases/smart-context.xml
--4.urlプロトコル、例えば、file:/D:/ALANWANG-AIA/Horse-workspace/chapter 3/target/clases/smart-context.xml
SpringはDefault Resource Loaderを提供しています。Default Resource Loaderは以上列挙した機能を実現した上で、開発者にカスタム拡張インターフェースProtocolverを提供しています。開発者はこのインターフェースのカスタマイズされた個性的なリソース表現を実現できます。コードは以下の通りです。
@Override
public Resource getResource(String location) {
Assert.notNull(location, "Location must not be null");
for (ProtocolResolver protocolResolver : this.protocolResolvers) { // 1
Resource resource = protocolResolver.resolve(location, this);
if (resource != null) {return resource;}
}
if (location.startsWith("/")) {return getResourceByPath(location);} //2
else if (location.startsWith(CLASSPATH_URL_PREFIX)) { //3
return new ClassPathResource(location.substring(CLASSPATH_URL_PREFIX.length()), getClassLoader());
}
else {
try {
// Try to parse the location as a URL...
URL url = new URL(location); //4
return new UrlResource(url);
}
catch (MalformedURLException ex) {
// No URL -> resolve as resource path.
return getResourceByPath(location); //5
}
}
}
ステップ1は、まず拡張プロトコルの解析器を用いてリソースアドレスを解析し、リターンする。例えば、リソース解析器をカスタマイズしてプレフィックス「クラスパス:」の解析を行うことができます。まずProtocoll Resoloverインターフェースを実現します。
class ClasspathPreProtocolResolver implements ProtocolResolver{
private static String CLASS_PATH_PRE="classpath:";
public Resource resolve(String location, ResourceLoader resourceLoader) {
if( location.startsWith(CLASS_PATH_PRE)) {
return new ClassPathResource(location.substring(CLASS_PATH_PRE.length()));
}
return null;
}
}
ステップ2において、locationがスラッシュで始まると仮定すると、このクラスのgetsResource ByPath(String path)方法を呼び出し、コードは以下の通りである。
protected Resource getResourceByPath(String path) {
return new ClassPathContextResource(path, getClassLoader());
}
ステップ3において、リソース表現がクラスパスで始まると、プレフィックスcasspathを除いたパスをカットし、Class PathResourceの構造パラメータとして、Class PathResourceのインスタンスを生成して返します。私たちはweb.xmlの中で次のように構成できます。
<context-param>
<param-name>contextConfigLocation</param-name>
<param-value>classpath:/config/applicationContext.xml</param-value>
</context-param>
6、refresh()
内部の実装により、refresh()
全体の方法は、Bean工場の初期化、post-processorの登録、呼び出し、beanの実装、イベントリリースなど、Spring容器全体の初期化とローディングのすべてのロジックを担っていることがほぼ理解できる。以上が本文の全部です。皆さんの勉強に役に立つように、私たちを応援してください。