原文地址:http://hi.baidu.com/hougbin/blog/item/fa93398b2a805b19c9fc7a7a.html
关于spring加载 xml文件早就思考过一段时间,主要矛盾集中在两点
1.b/s结构与c/s结构加载方式有和不同,如果在b/s中要用到c/s的加载方式如何实现最好
2.spring加载配置文件共有多少种方法
今天在jbpm的项目中,又遇到了这个问题,情况如下:建立一个工作流平台,b/s结构,spring+ext的结构。同事需要我写一个类,返回一些参数。由于需要这个类与servlet并无关系,再次考虑到在b/s结构中手动加载ApplicationContext
spring加载bean的方法总结:
方法1: 在初始化时保存ApplicationContext对象,可再分为如下三种
1.1 XmlBeanFactory 引用资源
Resource resource = new ClassPathResource("appcontext.xml");
BeanFactory factory = new XmlBeanFactory(resource);
1.2 二:ClassPathXmlApplicationContext 编译路径
ApplicationContext factory=new ClassPathXmlApplicationContext("classpath:appcontext.xml");
ApplicationContext factory=new ClassPathXmlApplicationContext("appcontext.xml"); // src目录下的
ApplicationContext factory=new ClassPathXmlApplicationContext("conf/appcontext.xml"); // src/conf 目录下的
ApplicationContext factory=new ClassPathXmlApplicationContext("file:G:/Test/src/appcontext.xml");
1.3 三 : 用文件系统的路径
ApplicationContext factory=new FileSystemXmlApplicationContext("src/appcontext.xml");
//使用了 classpath: 前缀,作为标志, 这样,FileSystemXmlApplicationContext 也能够读入classpath下的相对路径
ApplicationContext factory=new FileSystemXmlApplicationContext("classpath:appcontext.xml");
ApplicationContext factory=new FileSystemXmlApplicationContext("file:G:/Test/src/appcontext.xml");
ApplicationContext factory=new FileSystemXmlApplicationContext("G:/Test/src/appcontext.xml");
说明:这种方式适用于采用Spring框架的独立应用程序,需要程序通过配置文件手工初始化Spring的情况。
方法2 :通过Spring提供的工具类获取ApplicationContext对象
代码:
import org.springframework.web.context.support.WebApplicationContextUtils;
ApplicationContext ac1 = WebApplicationContextUtils.getRequiredWebApplicationContext(ServletContext sc);
ApplicationContext ac2 = WebApplicationContextUtils.getWebApplicationContext(ServletContext sc);
ac1.getBean("beanId");
ac2.getBean("beanId");
说明:
这种方式适合于采用Spring框架的B/S系统,通过ServletContext对象获取ApplicationContext对象,然后在通过它获取需要的类实例。
上面两个工具方式的区别是,前者在获取失败时抛出异常,后者返回null。
方法3: 继承自抽象类ApplicationObjectSupport
说明:抽象类ApplicationObjectSupport提供getApplicationContext()方法,可以方便的获取到ApplicationContext。
Spring初始化时,会通过该抽象类的setApplicationContext(ApplicationContext context)方法将ApplicationContext 对象注入。
方法4: 继承自抽象类WebApplicationObjectSupport
说明:类似上面方法,调用getWebApplicationContext()获取WebApplicationContext
方法5: 实现接口ApplicationContextAware
说明:实现该接口的setApplicationContext(ApplicationContext context)方法,并保存ApplicationContext 对象。
Spring初始化时,会通过该方法将ApplicationContext对象注入。
后种方法我没用过。最后使用的是:ApplicationContext ctx = new ClassPathXmlApplicationContext("classpath*:spring/applicationContext*.xml");
读取src/spring下的多个配置文件。
但是这样实际是加载两次配置文件,感觉不太好
网上这个文章,时间要试验一下
在web应用中一般用ContextLoaderListener加载webapplication,如果需要在action之外或者control类之外获取webapplication思路之一是,单独写个类放在static变量中,
类似于:
public class AppContext {
private static AppContext instance;
private AbstractApplicationContext appContext;
public synchronized static AppContext getInstance() {
if (instance == null) {
instance = new AppContext();
}
return instance;
}
private AppContext() {
this.appContext = new ClassPathXmlApplicationContext(
"/applicationContext.xml");
}
public AbstractApplicationContext getAppContext() {
return appContext;
}
}
不过这样,还是加载了2次applicationcontext,servlet一次,路径加载一次;觉得不如直接用路径加载,舍掉servlet加载
在网上也找了些其他说法:实现ApplicationContextAware,,, 接口,或者servletcontextAware接口,还要写配置文件。有的竟然要把配置文件里的listener,换成自己的类,这样纯粹多此一举。 不过有的应用不是替换,是在补一个listener,
我在一版的jpetstore(具体那一版不知道)里发现了这个:
[web.xml]里
<listener>
<listener-class>org.springframework.web.context.ContextLoaderListener</listener-class>
</listener>
<listener>
<listener-class>com.ibatis.jpetstore.util.SpringInit</listener-class>
</listener>其中SpringInit实现接口ServletContextListener :
package com.ibatis.jpetstore.util;
import javax.servlet.ServletContextEvent;
import javax.servlet.ServletContextListener;
import org.springframework.context.ApplicationContext;
import org.springframework.web.context.WebApplicationContext;
import org.springframework.web.context.support.WebApplicationContextUtils;
public class SpringInit implements ServletContextListener {
private static WebApplicationContext springContext;
public SpringInit() {
super();
}
public void contextInitialized(ServletContextEvent event) {
springContext = WebApplicationContextUtils.getWebApplicationContext(event.getServletContext());
}
public void contextDestroyed(ServletContextEvent event) {
}
public static ApplicationContext getApplicationContext() {
return springContext;
}
}
在其中的一个bean的构造里SpringInit获取applicationcontext,代码:
public OrderBean() {
this(
(AccountService) SpringInit.getApplicationContext().getBean("accountService"),
(OrderService) SpringInit.getApplicationContext().getBean("orderService") );
}
恩,这种在action,servlet之外的bean里获取applicationcontext的方法值得参考,应该有用
Spring中bean的实例化顺序
一.加载顺序:
先构造函数(无参的构造函数)——>然后是bean的set方法注入——>InitializingBean的afterPropertiesSet方法——>init-method方法
Spring装配Bean的过程
1. 实例化;
2. 设置属性值;
3. 如果实现了BeanNameAware接口,调用setBeanName设置Bean的ID或者Name;
4. 如果实现BeanFactoryAware接口,调用setBeanFactory 设置BeanFactory;
5. 如果实现ApplicationContextAware,调用setApplicationContext设置ApplicationContext
6. 调用BeanPostProcessor的预先初始化方法;
7. 调用InitializingBean的afterPropertiesSet()方法;
8. 调用定制init-method方法;
9. 调用BeanPostProcessor的后初始化方法;
Spring容器关闭过程
1. 调用DisposableBean的destroy();
2. 调用定制的destroy-method方法;
二. 实例代码:
1. 接口类:
public interface Axe {
String chop();
}
public interface Person {
void useAxe();
}
2.实现类:
public class SteelAxe implements Axe {
private int count;
public SteelAxe(){
System.out.println("SteelAxe的无参构造器被调用......");
}
public String chop() {
return "铁斧用了"+(++count)+"次";
}
}
public class American implements Person,InitializingBean {
private Axe axe;
private String name;
public American(){
System.out.println("American的无参构造器被调用.......");
}
public American(Axe axe){
this.axe=axe;
System.out.println("American的有参构造器被调用.......");
}
public void setAxe(Axe axe) {
System.out.println("Spring执行依赖关系注入......."+axe);
this.axe = axe;
}
public Axe getAxe() {
return axe;
}
public String getName() {
return name;
}
public void setName(String name) {
this.name = name;
}
public void useAxe() {
System.out.println("美国人使用:"+axe.chop());
System.out.println("参数name:"+name);
}
public void init(){ // 查看bean后处理器是在bean初始化之前还是之后起作用的呢,书写该
方法该类必须在配置市加上属性init-method="init",方法名是任意的,只要一致就行
System.out.println("正在执行初始化方法init......");
}
public void afterPropertiesSet() throws Exception { // 接口InitializingBean里的方法,在
调用BeanPostProcessor的预先初始化方法后执行
System.out.println("正在执行初始化方法afterPropertiesSet......");
}
}
3.bean后处理器类:
public class MyBeanPostProcessor implements BeanPostProcessor {
public Object postProcessBeforeInitialization(Object bean, String beanName)
throws BeansException {
System.out.println("处理之前的bean:"+bean);
System.out.println("bean后处理器在bean初始化之前对"+beanName+"进行增强处理!");
return bean;
}
public Object postProcessAfterInitialization(Object bean, String beanName)
throws BeansException {
System.out.println("处理之后的bean:"+bean);
System.out.println("bean后处理器在bean初始化之后对"+beanName+"进行增强处理!");
return bean;
}
}
4.xml配置文件类:
<?xml version="1.0" encoding="UTF-8"?>
<beans
xmlns="http://www.springframework.org/schema/beans"
xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"
xsi:schemaLocation="http://www.springframework.org/schema/beans
http://www.springframework.org/schema/beans/spring-beans-2.0.xsd">
<bean id="axe" class="common.SteelAxe" scope="prototype" />
<bean id="steelAxe" class="common.SteelAxe" scope="prototype" />
<bean id="American" class="common.American" init-method="init">
<property name="axe" ref="steelAxe" />
<property name="name" value="依赖注入的值" />
</bean>
<!--配置bean后处理器 -->
<bean id="beanPostProcessor" class="common.MyBeanPostProcessor" />
</beans>
5.测试类:
public class test {
public static void main(String[] args) {
ApplicationContext factory = new FileSystemXmlApplicationContext(
"C:/Documents and Settings/yejunjie/workspace/utilTool/utilTool/WEB- INF/conf/applicationContext_test.xml");
Person p = (Person)factory.getBean("American");
p.useAxe();
}
}
6.执行结果(后台打印:
American的无参构造器被调用.......
SteelAxe的无参构造器被调用......
处理之前的bean:common.SteelAxe@1a33d48
bean后处理器在bean初始化之前对steelAxe进行增强处理!
处理之后的bean:common.SteelAxe@1a33d48
bean后处理器在bean初始化之后对steelAxe进行增强处理!
Spring执行依赖关系注入.......common.SteelAxe@1a33d48
处理之前的bean:common.American@b2a2d8
bean后处理器在bean初始化之前对American进行增强处理!
正在执行初始化方法afterPropertiesSet......
正在执行初始化方法init......
处理之后的bean:common.American@b2a2d8
bean后处理器在bean初始化之后对American进行增强处理!
美国人使用:铁斧用了1次
参数name:依赖注入的值
注意:
<bean id="axe" class="common.SteelAxe" scope="prototype" />是不会先初始化