spring配置文件是用于指导Spring工厂进行Bean生产、依赖关系注入(装配)及Bean实例分发的"图纸"。Java EE程序员必须学会并灵活应用这份"图纸"准确地表达自己的"生产意图"。Spring配置文件是一个或多个标准的XML文档,applicationContext.xml是Spring的默认配置文件,当容器启动时找不到指定的配置文档时,将会尝试加载这个默认的配置文件。
下面列举的是一份比较完整的配置文件模板,文档中各XML标签节点的基本用途也给出了详细的解释,这些XML标签节点在后续的知识点中均会用到,熟练掌握了这些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.5.xsd">
- <!-- 定义使用C3P0连接池的数据源 -->
- <bean id="dataSource" class="com.mchange.v2.c3p0.ComboPooledDataSource">
- <!-- 指定连接数据库的JDBC驱动 -->
- <property name="driverClass">
- <value>com.mysql.jdbc.Driver</value>
- </property>
- <!-- 连接数据库所用的URL -->
- <property name="jdbcUrl">
- <value>jdbc:mysql://localhost:3306/eportal?useUnicode=
- true&characterEncoding=gbk</value>
- </property>
- <!-- 连接数据库的用户名 -->
- <property name="user">
- <value>root</value>
- </property>
- <!-- 连接数据库的密码 -->
- <property name="password">
- <value>root</value>
- </property>
- <!-- 设置数据库连接池的最大连接数 -->
- <property name="maxPoolSize">
- <value>20</value>
- </property>
- <!-- 设置数据库连接池的最小连接数 -->
- <property name="minPoolSize">
- <value>2</value>
- </property>
- <!-- 设置数据库连接池的初始化连接数 -->
- <property name="initialPoolSize">
- <value>2</value>
- </property>
- <!-- 设置数据库连接池的连接的最大空闲时间,单位为秒 -->
- <property name="maxIdleTime">
- <value>20</value>
- </property>
- </bean>
- <!-- 定义Hibernate的SessionFactory -->
- <bean id="sessionFactory"
- class="org.springframework.orm.
- hibernate3.LocalSessionFactoryBean">
- <!-- 依赖注入上面定义的数据源dataSource -->
- <property name="dataSource" ref="dataSource" />
- <!-- 注册Hibernate的ORM映射文件 -->
- <property name="mappingResources">
- <list>
- <value>com/eportal/ORM/News.hbm.xml</value>
- <value>com/eportal/ORM/Category.hbm.xml</value>
- <value>com/eportal/ORM/Memberlevel.hbm.xml</value>
- <value>com/eportal/ORM/Cart.hbm.xml</value>
- <value>com/eportal/ORM/Traffic.hbm.xml</value>
- <value>com/eportal/ORM/Newsrule.hbm.xml</value>
- <value>com/eportal/ORM/Merchandise.hbm.xml</value>
- <value>com/eportal/ORM/Admin.hbm.xml</value>
- <value>com/eportal/ORM/Orders.hbm.xml</value>
- <value>com/eportal/ORM/Cartselectedmer.hbm.xml</value>
- <value>com/eportal/ORM/Newscolumns.hbm.xml</value>
- <value>com/eportal/ORM/Member.hbm.xml</value>
- </list>
- </property>
- <!-- 设置Hibernate的相关属性 -->
- <property name="hibernateProperties">
- <props>
- <!-- 设置Hibernate的数据库方言 -->
- <prop key="hibernate.dialect">org.hibernate.dialect.MySQLDialect</prop>
- <!-- 设置Hibernate是否在控制台输出SQL语句,开发调试阶段通常设为true -->
- <prop key="show_sql">true</prop>
- <!-- 设置Hibernate一个提交批次中的最大SQL语句数 -->
- <prop key="hibernate.jdbc.batch_size">50</prop>
- <prop key="show_sql">50</prop>
- </props>
- </property>
- </bean>
- <!--定义Hibernate的事务管理器HibernateTransactionManager -->
- <bean id="transactionManager"
- class="org.springframework.orm.hibernate3.HibernateTransactionManager">
- <!-- 依赖注入上面定义的sessionFactory -->
- <property name="sessionFactory" ref="sessionFactory" />
- </bean>
- <!--定义Spring的事务拦截器TransactionInterceptor -->
- <bean id="transactionInterceptor"
- class="org.springframework.transaction.interceptor.TransactionInterceptor">
- <!-- 依赖注入上面定义的事务管理器transactionManager -->
- <property name="transactionManager" ref="transactionManager" />
- <!-- 定义需要进行事务拦截的方法及所采用的事务控制类型 -->
- <property name="transactionAttributes">
- <props>
- <!-- 以browse、list、load、get及is开头的所有方法采用只读型事务控制类型 -->
- <prop key="browse*">PROPAGATION_REQUIRED,readOnly</prop>
- <prop key="list*">PROPAGATION_REQUIRED,readOnly</prop>
- <prop key="load*">PROPAGATION_REQUIRED,readOnly</prop>
- <prop key="get*">PROPAGATION_REQUIRED,readOnly</prop>
- <prop key="is*">PROPAGATION_REQUIRED,readOnly</prop>
- <!-- 所有方法均进行事务控制,如果当前没有事务,则新建一个事务 -->
- <prop key="*">PROPAGATION_REQUIRED</prop>
- </props>
- </property>
- </bean>
- <!-- 定义BeanNameAutoProxyCreatorf进行Spring的事务处理 -->
- <bean
- class="org.springframework.aop.framework.autoproxy.
- BeanNameAutoProxyCreator">
- <!-- 针对指定的bean自动生成业务代理 -->
- <property name="beanNames">
- <list>
- <value>adminService</value>
- <value>columnsService</value>
- <value>newsService</value>
- <value>crawlService</value>
- <value>memberLevelService</value>
- <value>memberService</value>
- <value>categoryService</value>
- <value>merService</value>
- <value>cartService</value>
- <value>ordersService</value>
- <value>trafficService</value>
- </list>
- </property>
- <!-- 这个属性为true时,表示被代理的是目标类本身而不是目标类的接口 -->
- <property name="proxyTargetClass">
- <value>true</value>
- </property>
- <!-- 依赖注入上面定义的事务拦截器transactionInterceptor -->
- <property name="interceptorNames">
- <list>
- <value>transactionInterceptor</value>
- </list>
- </property>
- </bean>
- <!-- 装配通用数据库访问类BaseDAOImpl -->
- <bean id="dao" class="com.eportal.DAO.BaseDAOImpl">
- <property name="sessionFactory" ref="sessionFactory" />
- </bean>
- <!-- 部署系统用户管理业务逻辑组件AdminServiceImpl -->
- <bean id="adminService" class="com.eportal.service.AdminServiceImpl">
- <property name="dao" ref="dao" />
- </bean>
- <!-- 部署新闻栏目管理业务逻辑组件ColumnsServiceImpl -->
- <bean id="columnsService" class="com.eportal.service.ColumnsServiceImpl">
- <property name="dao" ref="dao" />
- </bean>
- <!-- 部署订单管理业务逻辑组件OrderServiceImpl -->
- <bean id="ordersService" class="com.eportal.service.OrderServiceImpl">
- <property name="dao" ref="dao" />
- </bean>
- <!-- 部署流量统计业务逻辑组件TrafficServiceImpl -->
- <bean id="trafficService" class="com.eportal.service.TrafficServiceImpl">
- <property name="dao" ref="dao" />
- </bean>
- <!-- 部署Struts 2负责系统用户管理的控制器AdminAction -->
- <bean id="adminAction" class="com.eportal.struts.action.
- AdminAction"
- scope="prototype">
- <property name="service" ref="adminService" />
- </bean>
- <!-- 部署Struts 2负责新闻栏目管理的控制器ColumnsAction -->
- <bean id="columnsAction" class="com.eportal.struts.action.
- ColumnsAction"
- scope="prototype">
- <property name="service" ref="columnsService" />
- </bean>
- <!-- 部署Struts 2负责新闻管理的控制器NewsAction -->
- <bean id="newsAction" class="com.eportal.struts.action.
- NewsAction"
- scope="prototype">
- <property name="service" ref="newsService" />
- <property name="columnsService" ref="columnsService" />
- </bean>
- <!-- 部署Struts 2负责新闻采集规则管理的控制器CrawlAction -->
- <bean id="crawlAction" class="com.eportal.struts.action.
- CrawlAction"
- scope="prototype">
- <property name="service" ref="crawlService" />
- <property name="columnsService" ref="columnsService" />
- </bean>
- </beans>
Spring ApplicationContext.xml配置的12个技巧
<bean id="orderService"
class="com.lizjason.spring.OrderService">
<property name="companyName">
<value>lizjason</value>
</property>
<constructor-arg>
<ref bean="orderDAO">
</constructor-arg>
</bean>
<bean id="orderService"
class="com.lizjason.spring.OrderService">
<property name="companyName"
value="lizjason"/>
<constructor-arg ref="orderDAO"/>
</bean>
<bean id="billingService"
class="com.lizjason.spring.BillingService">
<constructor-arg index="0" value="lizjason"/>
<constructor-arg index="1" value="100"/>
</bean>
<bean id="billingService"
class="com.lizjason.spring.BillingService">
<constructor-arg type="java.lang.String"
value="lizjason"/>
<constructor-arg type="int" value="100"/>
</bean>
<bean id="abstractService" abstract="true"
class="com.lizjason.spring.AbstractService">
<property name="companyName"
value="lizjason"/>
</bean>
<bean id="shippingService"
parent="abstractService"
class="com.lizjason.spring.ShippingService">
<property name="shippedBy" value="lizjason"/>
</bean>
6.
尽量使用
ApplicationContext
装配bean,而不是用import
像Ant脚本中imports一样,
Spring的import
元素对于模块化bean的装配非常有用,例如:
<beans>
<import resource="billingServices.xml"/>
<import resource="shippingServices.xml"/>
<bean id="orderService"
class="com.lizjason.spring.OrderService"/>
<beans>
<bean id="orderService"
class="com.lizjason.spring.OrderService"
dependency-check="objects">
<property name="companyName"
value="lizjason"/>
<constructor-arg ref="orderDAO"/>
</bean>
description
元素中加入描述信息。例如:
<beans>
<description>
This file defines billing service
related beans and it depends on
baseServices.xml,which provides
service bean templates...
</description>
...
</beans>
description
元
素的一个好处就是工具可以很容易的把描述信息从这个元素中提取出来。
10.
和team members沟通变更
当你修改java源码后,要确保更改了配置文件中的相应部分并把这个情况告知你的team members。XML配置文件也是代码,它们是程序的重要组成部分,但它们很难阅读和维护。大多数时间里,你需要同时看XML配置文件和java代码才能知道是怎么回事。
11.
setter
注入和构造函数注入,优先使用前者
<bean id="orderService"
class="com.lizjason.spring.OrderService">
<constructor-arg ref="orderDAO"/>
</bean>
<bean id="billingService"
class="com.lizjason.spring.BillingService">
<property name="billingDAO"
ref="billingDAO">
</bean>
BillingService
bean用了setter注入。构造函数注入可以确保bean正确地构建,但是setter注入更加的灵活和易于控制,特别是当class有多个属性并且它们中的一些是可选的情况是更是如此。
ApplicationContextEclipse and IntelliJ,java代码更加的易于阅读,维护和管理比使XML文件
可以替你创建
java
对象
,
但不是所有的
java
对象都应该通过注入创建。例如
,
域对象就不应该通过
ApplicationContext
创建。Spring是一个优秀的框架,但是考虑到可读性和可操控性,基于XML配置的配置会在定义很多bean的时候出现麻烦。过渡使用依赖注入将会使XML配置更加的复杂和冗长。切记,当使用高效的IDE时,例如
<bean
id="beanId"(1)
name="beanName"(2)
class="beanClass"(3)
parent="parentBean"(4)
abstract="true | false"(5)
singleton="true | false"(6)
lazy-init="true | false | default"(7)
autowire="no | byName | byType | constructor | autodetect | default"(8)
dependency-check = "none | objects | simple | all | default"(9)
depends-on="dependsOnBean"(10)
init-method="method"(11)
destroy-method="method"(12)
factory-method="method"(13)
factory-bean="bean">(14)
</bean>
(1)、id: Bean的唯一标识名。它必须是合法的XML ID,在整个XML文档中唯一。
(2)、name: 用来为id创建一个或多个别名。它可以是任意的字母符合。多个别名之间用逗号或空格分开。
(3)、class: 用来定义类的全限定名(包名+类名)。只有子类Bean不用定义该属性。
(4)、parent: 子类Bean定义它所引用它的父类Bean。这时前面的class属性失效。子类Bean会继承父类Bean的所有属性,子类Bean也可以覆盖父类Bean的属性。注意:子类Bean和父类Bean是同一个Java类。
(5)、abstract(默认为”false”):用来定义Bean是否为抽象Bean。它表示这个Bean将不会被实例化,一般用于父类Bean,因为父类Bean主要是供子类Bean继承使用。
(6)、singleton(默认为“true”):定义Bean是否是Singleton(单例)。如果设为“true”,则在BeanFactory作用范围内,只维护此Bean的一个实例。如果设为“flase”,Bean将是Prototype(原型)状态,BeanFactory将为每次Bean请求创建一个新的Bean实例。
(7)、lazy-init(默认为“default”):用来定义这个Bean是否实现懒初始化。如果为“true”,它将在BeanFactory启动时初始化所有的Singleton Bean。反之,如果为“false”,它只在Bean请求时才开始创建Singleton Bean。
(8)、autowire(自动装配,默认为“default”):它定义了Bean的自动装载方式。
1、“no”:不使用自动装配功能。
2、“byName”:通过Bean的属性名实现自动装配。
3、“byType”:通过Bean的类型实现自动装配。
4、“constructor”:类似于byType,但它是用于构造函数的参数的自动组装。
5、“autodetect”:通过Bean类的反省机制(introspection)决定是使用“constructor”还是使用“byType”。
(9)、dependency-check(依赖检查,默认为“default”):它用来确保Bean组件通过JavaBean描述的所以依赖关系都得到满足。在与自动装配功能一起使用时,它特别有用。
1、 none:不进行依赖检查。
2、 objects:只做对象间依赖的检查。
3、 simple:只做原始类型和String类型依赖的检查
4、 all:对所有类型的依赖进行检查。它包括了前面的objects和simple。
(10)、depends-on(依赖对象):这个Bean在初始化时依赖的对象,这个对象会在这个Bean初始化之前创建。
(11)、init-method:用来定义Bean的初始化方法,它会在Bean组装之后调用。它必须是一个无参数的方法。
(12)、destroy-method:用来定义Bean的销毁方法,它在BeanFactory关闭时调用。同样,它也必须是一个无参数的方法。它只能应用于singleton Bean。
(13)、factory-method:定义创建该Bean对象的工厂方法。它用于下面的“factory-bean”,表示这个Bean是通过工厂方法创建。此时,“class”属性失效。
(14)、factory-bean:定义创建该Bean对象的工厂类。如果使用了“factory-bean”则“class”属性失效。
下面列出<ref>元素的所有可用的指定方式:
bean:可以在当前文件中查找依赖对象,也可以在应用上下文(ApplicationContext)中查找其它配置文件的对象。
local:只在当前文件中查找依赖对象。这个属性是一个XML IDREF,所以它指定的对象必须存在,否则它的验证检查会报错。
external:在其它文件中查找依赖对象,而不在当前文件中查找。
总的来说,<ref bean="..."/>和<ref local="..."/>大部分的时候可以通用。“bean”是最灵活的方式,它允许你在多个文件之间共享Bean。而“local”则提供了便利的XML验证。
getBean()
方法)都会产生一个新的bean实例,相当与一个new的操作,对于prototype作用域的bean,有一点非常重要,那就是Spring不能对一个prototype bean的整个生命周期负责,容器在初始化、配置、装饰或者是装配完一个prototype实例后,将它交给客户端,随后就对该prototype实例不闻不问了。不管何种作用域,容器都会调用所有对象的初始化生命周期回调方法,而对prototype而言,任何配置好的析构生命周期回调方法都将不会被调用。 清除prototype作用域的对象并释放任何prototype bean所持有的昂贵资源,都是客户端代码的职责。(让Spring容器释放被singleton作用域bean占用资源的一种可行方式是,通过使用 bean的后置处理器,该处理器持有要被清除的bean的引用。)
...
< listener >
< listener-class >org.springframework.web.context.request.RequestContextListener </ listener-class >
</ listener >
...
</ web-app >
,如果是Servlet2.4以前的web容器,那么你要使用一个javax.servlet.Filter的实现:
..
< filter >
< filter-name >requestContextFilter </ filter-name >
< filter-class >org.springframework.web.filter.RequestContextFilter </ filter-class >
</ filter >
< filter-mapping >
< filter-name >requestContextFilter </ filter-name >
< url-pattern >/* </ url-pattern >
</ filter-mapping >
...
</ web-app >
接着既可以配置bean的作用域了:
privatefinal ThreadLocal threadScope = new ThreadLocal() {
protected Object initialValue() {
returnnew HashMap();
}
};
public Object get(String name, ObjectFactory objectFactory) {
Map scope = (Map) threadScope.get();
Object object = scope.get(name);
if(object== null) {
object = objectFactory.getObject();
scope.put(name, object);
}
return object;
}
public Object remove(String name) {
Map scope = (Map) threadScope.get();
return scope.remove(name);
}
publicvoid registerDestructionCallback(String name, Runnable callback) {
}
public String getConversationId() {
// TODO Auto-generated method stub
returnnull;
}
}