- 浏览: 167434 次
- 性别:
- 来自: 北京
最新评论
-
TonyLee0329:
求真实案例
java中compareTo比较两个日期大小 -
fdyo3:
高手
两个值相同的Integer类型用!=比较出错的问题 -
宋小寒:
http://www.blogjava.net/sternin ...
java多线程socket通信---Telnet
spring的应用初始化流程一直没有搞明白,刚刚又碰到了相关的问题。决定得好好看看这个流程。我们在开发spring的项目当中基本上都会在web.xml通过:
[html] view plaincopyprint?
<context-param>
<param-name>contextConfigLocation</param-name>
<param-value>
/WEB-INF/conf/application-*.xml
</param-value>
</context-param>
来初始化各个spring的配置文件,但是我们只是知道这段代码的功能, 并不是很清楚我们配置了这段代码之后为什么就能去初始化配置文件。当然我们还会加上:
[html] view plaincopyprint?
<listener>
<listener-class>org.springframework.web.context.ContextLoaderListener</listener-class>
</listener>
这一个listener,我首先就会想contextConfigLocation这个一定能在ContextLoaderListener这个类当中找到,打开了源码,这个listener是实现了ServletContextListener这个接口的,这个接口只有两个方法:
[html] view plaincopyprint?
public interface ServletContextListener
extends EventListener
{
public abstract void contextInitialized(ServletContextEvent servletcontextevent);
public abstract void contextDestroyed(ServletContextEvent servletcontextevent);
}
而且它是继承了EventListener这个接口的,打开这个接口的代码让我大吃一惊,里面没有方法啥都没有:
[html] view plaincopyprint?
package java.util;
public interface EventListener
{
}
而且还是java.util包下的,并不是spring之中的东西。
这样找了之后没有找到,往回退到ContextLoaderListener这个类的方法上,contextInitialized方法是用来初始化上下文的:
[html] view plaincopyprint?
public void contextInitialized(ServletContextEvent event)
{
contextLoader = createContextLoader();
contextLoader.initWebApplicationContext(event.getServletContext());
}
方法中有个createContextLoader方法:
[html] view plaincopyprint?
protected ContextLoader createContextLoader()
{
return new ContextLoader();
}
这个方法返回了一个ContextLoader实例,进入到ContextLoader类中,按ctrl+f来寻找contextConfigLocation,这时没有出现电脑的咚的声音,找到了它:
[html] view plaincopyprint?
protected WebApplicationContext createWebApplicationContext(ServletContext servletContext, ApplicationContext parent)
throws BeansException
{
Class contextClass = determineContextClass(servletContext);
if(!(org.springframework.web.context.ConfigurableWebApplicationContext.class).isAssignableFrom(contextClass))
{
throw new ApplicationContextException("Custom context class [" + contextClass.getName() + "] is not of type [" + (org.springframework.web.context.ConfigurableWebApplicationContext.class).getName() + "]");
} else
{
ConfigurableWebApplicationContext wac = (ConfigurableWebApplicationContext)BeanUtils.instantiateClass(contextClass);
wac.setParent(parent);
wac.setServletContext(servletContext);
wac.setConfigLocation(servletContext.getInitParameter("<span style="color:#ff0000;">contextConfigLocation</span>"));
customizeContext(servletContext, wac);
wac.refresh();
return wac;
}
}
通过代码,ConfigurableWebApplicationContext设置了从servletContext获取到的参数的值,再进入ConfigurableWebApplicationContext的代码中,它只是一个接口,进入StaticWebApplicationContext的setConfigLocation方法:
[html] view plaincopyprint?
public void setConfigLocation(String configLocation)
{
if(configLocation != null)
throw new UnsupportedOperationException("StaticWebApplicationContext does not support config locations");
else
return;
}
这个方法中很奇怪,当参数不为空就抛出异常,查看spring的文档:The StaticWebApplicationContext class does not support this method.说是此类不支持这个方法,这下子又卡住了。又要退回去,看这句:
[html] view plaincopyprint?
ConfigurableWebApplicationContext wac = (ConfigurableWebApplicationContext)BeanUtils.instantiateClass(contextClass);
spring使用BeanUtils来初始化contextClass这个类实例,contextClass是通过以下代码得到的:
[html] view plaincopyprint?
protected Class determineContextClass(ServletContext servletContext)
throws ApplicationContextException
{
String contextClassName = servletContext.getInitParameter("contextClass");
if(contextClassName != null)
try
{
return ClassUtils.forName(contextClassName);
}
catch(ClassNotFoundException ex)
{
throw new ApplicationContextException("Failed to load custom context class [" + contextClassName + "]", ex);
}
contextClassName = defaultStrategies.getProperty((org.springframework.web.context.WebApplicationContext.class).getName());
try
{
return ClassUtils.forName(contextClassName, (org.springframework.web.context.ContextLoader.class).getClassLoader());
}
catch(ClassNotFoundException ex)
{
throw new ApplicationContextException("Failed to load default context class [" + contextClassName + "]", ex);
}
}
这里使用了反射,再来看BeanUtils的instantiateClass方法:
[html] view plaincopyprint?
return instantiateClass(clazz.getDeclaredConstructor((Class[])null), null);
通过反射得到contextClass的构造方法。下面是instantiateClass方法的重载,主要是下面两句代码:
[html] view plaincopyprint?
ReflectionUtils.makeAccessible(ctor);
return ctor.newInstance(args);
ctor是通过反射得到的contextClass的构造方法,args是构造方法当中的参数。这里为null,说明new了contextClass的无参构造方法。
这时又要退回到determineContextClass 这个方法中,我们主要看:
[html] view plaincopyprint?
contextClassName = defaultStrategies.getProperty((org.springframework.web.context.WebApplicationContext.class).getName());
这句代码,我们可以猜它是通过Properties的getProperty方法得到WebApplicationContext 的实例,这时我们又到了WebApplicationContext 这个接口当中,这个接口继承了ApplicationContext这个接口,我们都知道我们进行spring开发都会通过Application ctx=new FileSystemXmlApplicationContext("beans.xml");或ApplicationContext ctx=new ClassPathXmlApplicationContext("beans.xml");或ServletContext servletContext = request.getSession().getServletContext();ApplicationContext ctx = WebApplicationContextUtils.getWebApplicationContext(servletContext);这三种方法获得一个ApplicationContext,然后就可以对配置文件当中的bean进行操作了。所以这里我们基本上已经搞清楚初始化spring配置文件的流程了。
总结:通过查看这几个类的源代码,java的反射使用范围之广再次体现出来。如看了之后觉得有错误或者不同意见,欢迎提出来,我也是第一次才研究这个问题。
[html] view plaincopyprint?
<context-param>
<param-name>contextConfigLocation</param-name>
<param-value>
/WEB-INF/conf/application-*.xml
</param-value>
</context-param>
来初始化各个spring的配置文件,但是我们只是知道这段代码的功能, 并不是很清楚我们配置了这段代码之后为什么就能去初始化配置文件。当然我们还会加上:
[html] view plaincopyprint?
<listener>
<listener-class>org.springframework.web.context.ContextLoaderListener</listener-class>
</listener>
这一个listener,我首先就会想contextConfigLocation这个一定能在ContextLoaderListener这个类当中找到,打开了源码,这个listener是实现了ServletContextListener这个接口的,这个接口只有两个方法:
[html] view plaincopyprint?
public interface ServletContextListener
extends EventListener
{
public abstract void contextInitialized(ServletContextEvent servletcontextevent);
public abstract void contextDestroyed(ServletContextEvent servletcontextevent);
}
而且它是继承了EventListener这个接口的,打开这个接口的代码让我大吃一惊,里面没有方法啥都没有:
[html] view plaincopyprint?
package java.util;
public interface EventListener
{
}
而且还是java.util包下的,并不是spring之中的东西。
这样找了之后没有找到,往回退到ContextLoaderListener这个类的方法上,contextInitialized方法是用来初始化上下文的:
[html] view plaincopyprint?
public void contextInitialized(ServletContextEvent event)
{
contextLoader = createContextLoader();
contextLoader.initWebApplicationContext(event.getServletContext());
}
方法中有个createContextLoader方法:
[html] view plaincopyprint?
protected ContextLoader createContextLoader()
{
return new ContextLoader();
}
这个方法返回了一个ContextLoader实例,进入到ContextLoader类中,按ctrl+f来寻找contextConfigLocation,这时没有出现电脑的咚的声音,找到了它:
[html] view plaincopyprint?
protected WebApplicationContext createWebApplicationContext(ServletContext servletContext, ApplicationContext parent)
throws BeansException
{
Class contextClass = determineContextClass(servletContext);
if(!(org.springframework.web.context.ConfigurableWebApplicationContext.class).isAssignableFrom(contextClass))
{
throw new ApplicationContextException("Custom context class [" + contextClass.getName() + "] is not of type [" + (org.springframework.web.context.ConfigurableWebApplicationContext.class).getName() + "]");
} else
{
ConfigurableWebApplicationContext wac = (ConfigurableWebApplicationContext)BeanUtils.instantiateClass(contextClass);
wac.setParent(parent);
wac.setServletContext(servletContext);
wac.setConfigLocation(servletContext.getInitParameter("<span style="color:#ff0000;">contextConfigLocation</span>"));
customizeContext(servletContext, wac);
wac.refresh();
return wac;
}
}
通过代码,ConfigurableWebApplicationContext设置了从servletContext获取到的参数的值,再进入ConfigurableWebApplicationContext的代码中,它只是一个接口,进入StaticWebApplicationContext的setConfigLocation方法:
[html] view plaincopyprint?
public void setConfigLocation(String configLocation)
{
if(configLocation != null)
throw new UnsupportedOperationException("StaticWebApplicationContext does not support config locations");
else
return;
}
这个方法中很奇怪,当参数不为空就抛出异常,查看spring的文档:The StaticWebApplicationContext class does not support this method.说是此类不支持这个方法,这下子又卡住了。又要退回去,看这句:
[html] view plaincopyprint?
ConfigurableWebApplicationContext wac = (ConfigurableWebApplicationContext)BeanUtils.instantiateClass(contextClass);
spring使用BeanUtils来初始化contextClass这个类实例,contextClass是通过以下代码得到的:
[html] view plaincopyprint?
protected Class determineContextClass(ServletContext servletContext)
throws ApplicationContextException
{
String contextClassName = servletContext.getInitParameter("contextClass");
if(contextClassName != null)
try
{
return ClassUtils.forName(contextClassName);
}
catch(ClassNotFoundException ex)
{
throw new ApplicationContextException("Failed to load custom context class [" + contextClassName + "]", ex);
}
contextClassName = defaultStrategies.getProperty((org.springframework.web.context.WebApplicationContext.class).getName());
try
{
return ClassUtils.forName(contextClassName, (org.springframework.web.context.ContextLoader.class).getClassLoader());
}
catch(ClassNotFoundException ex)
{
throw new ApplicationContextException("Failed to load default context class [" + contextClassName + "]", ex);
}
}
这里使用了反射,再来看BeanUtils的instantiateClass方法:
[html] view plaincopyprint?
return instantiateClass(clazz.getDeclaredConstructor((Class[])null), null);
通过反射得到contextClass的构造方法。下面是instantiateClass方法的重载,主要是下面两句代码:
[html] view plaincopyprint?
ReflectionUtils.makeAccessible(ctor);
return ctor.newInstance(args);
ctor是通过反射得到的contextClass的构造方法,args是构造方法当中的参数。这里为null,说明new了contextClass的无参构造方法。
这时又要退回到determineContextClass 这个方法中,我们主要看:
[html] view plaincopyprint?
contextClassName = defaultStrategies.getProperty((org.springframework.web.context.WebApplicationContext.class).getName());
这句代码,我们可以猜它是通过Properties的getProperty方法得到WebApplicationContext 的实例,这时我们又到了WebApplicationContext 这个接口当中,这个接口继承了ApplicationContext这个接口,我们都知道我们进行spring开发都会通过Application ctx=new FileSystemXmlApplicationContext("beans.xml");或ApplicationContext ctx=new ClassPathXmlApplicationContext("beans.xml");或ServletContext servletContext = request.getSession().getServletContext();ApplicationContext ctx = WebApplicationContextUtils.getWebApplicationContext(servletContext);这三种方法获得一个ApplicationContext,然后就可以对配置文件当中的bean进行操作了。所以这里我们基本上已经搞清楚初始化spring配置文件的流程了。
总结:通过查看这几个类的源代码,java的反射使用范围之广再次体现出来。如看了之后觉得有错误或者不同意见,欢迎提出来,我也是第一次才研究这个问题。
发表评论
-
在同一台服务器上配置多个Tomcat
2015-08-19 18:00 740在一台服务器上配置 ... -
Mybatis MapperScannerConfigurer 自动扫描 将Mapper接口生成代理注入到Spring
2015-08-19 14:25 1200Mybatis MapperScannerConfig ... -
jboss之启动加载过程详解(-)
2015-05-18 15:06 1204今天看了看jboss的boot.l ... -
Mybatis中几个重要类
2015-04-24 18:44 1319本文基于Mybatis3.2.0版本的代码。 1.org. ... -
『转』Spring Security的核心拦截器
2014-08-07 11:36 9531. HttpSessionContextIntegrat ... -
quartz CronExpression表达式
2014-07-24 14:33 546一个cron表达式有至少6 ... -
environments was not found on the java.library.path: E:\Program Files (x86)\Java
2014-07-23 11:10 916启动的时候出现的这个信息,不是错误,程序也可以正常的运行, ... -
java String 以及字符串直接量 与 字符串驻留池 ...
2014-07-16 13:56 933字符串 (String) 是 java 编程语言中的核心类 ... -
hibernate缓存机制
2014-07-03 00:02 751http://www.blogjava.net/tbwshc/ ... -
hibernate修改部分字段
2014-07-02 23:39 760http://blog.csdn.net/kkdelta/ar ... -
解惑 spring 嵌套事务
2014-06-24 15:44 711/** * @author 王政 * @dat ... -
浅析Java虚拟机结构与机制
2014-05-07 17:04 530浅析Java虚拟机结构与机制 本文旨在给所有希望了解J ... -
多语言调用之 Java调用C/C++
2014-04-09 14:52 6861.创建一个类com.test.TestCall内容如下 ... -
多线程实例
2014-04-09 14:47 530编写具有多线程能力的程序经常会用到的方法有: r ... -
java中的native关键字
2014-04-09 11:43 732JNI是Java Native Interface的 ... -
Spring分布式事务在service中动态切换数据源
2014-04-02 18:23 1133项目采用的是struts2+spring+ibatis架构, ... -
java的finalize
2014-03-31 14:04 518目录 基本预备相关知 ... -
读懂tomcat6 之Catalina.sh --注释版
2014-03-24 10:27 1163粗体字部分是我的注释,可能对那些不太熟悉Shell 的人有 ... -
tomcat下使用cronolog对catalina.out日志文件分割
2014-03-24 10:24 874tomcat 的catalina.out文件的不断 ... -
jsp直接使用session
2014-03-12 11:07 721在servlet中,要得到session并设值 要用 requ ...
相关推荐
TM4C123系列是德州仪器(TI)推出的一款基于ARM Cortex-M4内核的微控制器,广泛应用于工业控制、嵌入式系统和电子设计大赛等场合。这个压缩包"TM4C123常用代码模板.zip"包含了针对TM4C123的源码示例和工程模板,对...
mpg123是一款开源的音频播放器,主要设计用于高效地播放MP3音频格式的文件。这个"mpg123 windows版本"是专为Windows操作系统编译的,为用户提供了在Windows环境下轻松享受音乐的工具。由于其轻量级的特性,mpg123在...
libmpg123是一个C++编写的库,专用于MP3音频的解码,它支持多种平台,包括Android。本篇文章将深入探讨libmpg123库在Android armeabi-v7a架构上的应用及实现音频解码的相关知识。 libmpg123库提供了高效且稳定的MP3...
### 74HC123 双可重触发单稳态触发器详解 #### 基本概述 74HC123是一款双通道可重触发的单稳态触发器,具备清除端功能,适用于各种电子电路设计中对脉冲宽度进行精确控制的需求。该器件有两种类型:CT54123/CT74123...
《OpenDSS123节点源码解析:深入理解配电网潮流计算》 OpenDSS(Open Source Distribution System Simulator)是一款强大的电力系统分布式仿真软件,主要用于配电网的模拟与分析。在电力工程领域,OpenDSS被广泛...
标题中的"**url.hao123.aspjzy(1).rar_123.byios1:59898_www.yuecai.com123**"暗示这是一个关于hao123网站源代码的压缩文件,其中可能包含了一些ASP(Active Server Pages)文件和其他相关资源。"123.byios1:59898...
### 74LS123 双可重触发单稳态触发器详解 #### 基本概述 74LS123是一款双可重触发单稳态触发器芯片,带有清除端,常用于产生精确的脉冲信号。该芯片有两个独立的单稳态触发器单元,每个单元都可以通过外部电阻和...
libmpg123库是一款专为C++编程语言设计的高效音频解码库,主要用于处理MP3格式的音频文件。在Windows x64操作系统环境下,它提供了强大的功能,使开发者能够轻松地在64位系统上实现MP3音频的解码和播放。在本文中,...
### 一、123资源最新采集接口概述 123资源最新采集接口主要涉及一系列用于不同内容管理系统(Content Management System,简称CMS)的数据抓取接口。这些接口能够帮助用户快速地获取视频资源和其他媒体内容,为用户...
【标题】"hao123网站源码"所涉及的知识点主要集中在网站开发、前端技术、网页设计以及用户体验方面。hao123作为中国知名的网址导航网站,它的源码解析对于我们理解互联网产品的构建、优化和运营具有重要的学习价值。...
74ls123引脚及功能 74ls123功能表 说明:1.外接电容接在 Cext(正)和 Rext/Cext(正)之间 2.为了改善脉冲宽度的精度和重复性,可在 Rext/Cext 和 Vcc 之间接外接电阻。 3.为了得到可变脉冲宽度,可在 ...
### LV123标准知识点详解 #### 一、概述 LV123是戴姆勒股份公司(Daimler AG)发布的关于道路车辆高压组件电气特性和电气安全的标准。该标准详细规定了高压组件在设计、生产和测试过程中的具体要求与测试方法。LV...
《123网盘在线解析源码:深入理解与应用》 在当今互联网时代,网盘已经成为用户存储、分享和访问文件的重要工具。123网盘在线解析源码是针对123网盘服务的一种技术实现,它允许用户通过浏览器直接访问和下载存储在...
FLUKE 123 示波表是一款专为电气工程师和维修技术人员设计的便携式测量工具,它结合了示波器和万用表的功能,能够帮助用户在各种环境中进行故障诊断和性能评估。这份“FLUKE 123 示波表 维修资料”包含了重要的技术...
在电子工程领域,74LS123是一种常用的可重触发集成单稳态触发器,它在各种定时和脉冲整形应用中发挥着关键作用。这个实验电路的Multisim源文件提供了对74LS123工作原理的直观理解和实践操作的机会。Multisim是一款...
123网盘解析PHP版本源码是一种非常实用的工具,可以方便地帮助用户在网页上直接解析出其它网站中的资源,并提供下载链接。当用户需要获取某些资源时,往往需要通过各种搜索引擎或者专业的资源网站进行查找,而且很多...
在IT行业中,网络存储服务,如123网盘和cc网盘,为用户提供便捷的云存储和分享功能。"123网盘cc网盘快捷下载工具"是一款旨在优化和加速从这两个平台下载文件的软件。这类工具通常通过集成特定的下载技术,帮助用户绕...
单文件123网盘在线解析PHP源码,一种非常实用的工具,可 以方便地帮助用户在网页上直接解析出其它网站中的资源,并 提供下载链接。当用户需要获取某些资源时,往往需要通过各 种搜索引擎或者专业的资源网站进行查找...
《Hans123Trader_v9_02:外汇市场的突破交易策略详解》 Hans123Trader_v9_02,这是一个在外汇交易领域备受推崇的交易策略,其核心在于利用特定的突破线来识别市场趋势并进行交易决策。这个版本,Hans123-9.02,是对...
【PHP123源码2.0】是一个针对PHP编程语言的开源项目,它提供了丰富的功能和优化,旨在帮助开发者更高效地构建Web应用程序。PHP123源码的更新到2.0版本,通常意味着在性能、稳定性和功能方面都有所提升。"飞机开绿灯...