spring 加载配置文件


原文地址: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" />是不会先初始化

  • 0
    点赞
  • 1
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值