事务:要么全有要么全无的操作叫做事务
事务的几个特征:
(1)原子性Atomic
(2)一致性Consistent
(3)隔离性Isolated
(4)持久性Durable
Spring可供选择的事务管理器
(1)DataSourceTransactionManager在单一的jdbc datasource中管理事务,配置bean时注入DataSource
(2)HibernateTrancationManager,配置bean时注入SessionFactory,
HibernateTrancationManager把事务管理委托给从Hibernate session中获得的net.sf.hibernate.Trancation对象
(2)JdoTrancationManager
(2)JtaTrancationManager一个事务跨越多个资源时必须使用,它把事务管理委托给一个JTA的实现,配置Bean时注入trancationManagerName,通过JNDI找到该事务管理器,它和javax.trancation.UserTrancation以及javax.tranction.TrancationManager对象一起工作,事务成功调用UserTrancation.commit(),否则调用UserTrancation.rollback()方法
(2)PersistenceBrokerTrancationManager
Spring中编写事务
(1)编程实现
使用Spring的TrancationTemplate类
transactionTemplate通过注入来获得
(2)声明式实现
它是通过AOP框架实现的,具体配置过程:
事务属性
Spring中事务属性是对事务策略如何应用到方法的描述,这个描述包括下列参数:
传播行为
定义了关于客户端和被调用方法的事务边界,Spring中有7种不同的值
隔离级别
多个事务同时访问一个数据,就会造成脏读dirty read,不可重复读nonrepeatedable read,幻读phantom read
Spring有5种隔离级别
只读提示
如果一个事务只对数据库执行读操作,可以声明这个事务为只读,数据库就可以应用一些优化措施。
只读的优化措施是在事务启动时有后端的数据库实施的,所以把只有那些有可能启动新事务的传播行为标示为只读才有意义。
此外,如果使用hibernate作为持久化机制,声明一个只读事务将会把hibernate的flush设置为flush_never,避免和数据库进行不比要得同步,将所有的更新延迟到事务的结束。
事务超时间隔
在那些可能启动新事务的传播行为上设置事务超时间隔才有意义。
声明一个简单的事务策略
TransactionProxyFactoryBean 有一个transactionAttributeSource属性,这个属性被设置成TransactionAttributeSource的实例,TransactionProxyFactoryBean 是作为在方法上查找事务属性的一个参考
TransactionAttributeSource的接口定义
随着transactionAttributeSource的bean对象的声明,所有被TransactionProxyFactoryBean的目标类代理的方法都被执行到一个事务环境中了,但没有指定哪个方法是事务性的,这个由MatchAlwaysTransactionAttributeSource来指定,它是TransactionAttributeSource的一个最简单的实现,每次调用,都是返回相同的TransactionAttribute [required和isolation_default]
可以通过配置一个DefaultTransactionAttribute的bean,来修改MatchAlwaysTransactionAttributeSource默认的事务属性
(1)配置一个DefaultTransactionAttribute
(2)把myDefautAttribute注入到MatchAlwaysTransactionAttributeSource的transactionAttribute属性中即可
对目标对象的每个方法实施不同的事务管理
使用NameMatchTransactionAttributeSource
NameMatchTransactionAttributeSource可以让你在方法上配置不同的事务,配置一个它的实例bean,并注入到TransactionProxyFactoryBean中的transactionAttributeSource中
NameMatchTransactionAttributeSource实例配置
不使用NameMatchTransactionAttributeSource
TransactionProxyFactoryBean中有一个transactionProperties属性,我们可以通过配置这个属性来完成不同方法不同事务的配置
用元数据声明事务 page 193
bean的配置
我们通过修改transactionAttributeSource的注入替换为注入AttributesTransactionAtrributeSource即可,下面是AttributesTransactionAtrributeSource的配置
java方法上加上事务标签
在类或者方法前面的注释中增加doclet标签,标签有以下几种形式
修剪事务声明
繁复的XML配置文件
当你选择了TransactionAttributeSource,并且把服务层的方法声明为事务的,并且定义了合适的事务管理器,并完成了如下bean的配置
浏览这个配置对象,你会发现好几对服务/目标,你会发现好几个对象的声明,它们的名字暗示它们是服务对象比如txLoanBS,但是它只是TransactionProxyFactoryBean的一个实例,真正的服务对象是被命名为带target后缀并被注入到TransactionProxyFactoryBean中target属性中的bean
一般项目中服务对象都用了同样的方法定义,并设定了相同的事务管理器相同的事务属性源,看上去是一大堆繁复的XML文件
消除繁复的XML配置
Spring提供了2种方法:
(1)bean继承,从父TransactionProxyFactoryBean继承,使用Spring对父bean的支持,即bean的parent属性继承父bean的属性
对于上面的bean的具体实现过程如下
首先通过lazy-init=true,它告诉容器不要创建bean,构造一个抽象的TransactionProxyFactoryBean Bean
然后再配置某个具体的事务服务
<bean id="txLoanBS" parent="abstractTxDefinition">
<property name="proxyInterfaces">
<list>
<value>cn.ccb.csps.efbuss.bizservice.ITxLoanBS</value>
</list>
</property>
<property name="target">
<bean id="txLoanBSImplTarget"
class="cn.ccb.csps.efbuss.bizservice.impl.TxLoanBSImpl">
</property>
</bean>
(2)AOP自动代理
首先声明一个bean,成为DefaultAdvisorAutoProxyCreator的实列
DefaultAdvisorAutoProxyCreator将在context中遍历advisor(对于事务为TransactionAttributeSourceAdvisor)自动用它们来代理匹配advisor的pointcut的所有bean
当用自动代理的时候,事务属性源采用AttributesTransactionAtrributeSource是最好的选择了,使方法事务化和非事务化,只不过是往方法中添加合适的元数据就可以了
事务的几个特征:
(1)原子性Atomic
(2)一致性Consistent
(3)隔离性Isolated
(4)持久性Durable
Spring可供选择的事务管理器
(1)DataSourceTransactionManager在单一的jdbc datasource中管理事务,配置bean时注入DataSource
(2)HibernateTrancationManager,配置bean时注入SessionFactory,
HibernateTrancationManager把事务管理委托给从Hibernate session中获得的net.sf.hibernate.Trancation对象
(2)JdoTrancationManager
(2)JtaTrancationManager一个事务跨越多个资源时必须使用,它把事务管理委托给一个JTA的实现,配置Bean时注入trancationManagerName,通过JNDI找到该事务管理器,它和javax.trancation.UserTrancation以及javax.tranction.TrancationManager对象一起工作,事务成功调用UserTrancation.commit(),否则调用UserTrancation.rollback()方法
(2)PersistenceBrokerTrancationManager
Spring中编写事务
(1)编程实现
使用Spring的TrancationTemplate类
transactionTemplate.execute( new TransactionCallback(){ public Object doInHibernate(TransactionStatus ts){ try{ //你的数据库操作代码 }catch(Exception e){ // 出现异常则回滚 ts.setRollbackOnly(); } //如果成功则提交 return null; } });
transactionTemplate通过注入来获得
<bean id="transactionTemlate" class="*****TrancationTemplate"> <property name="transactionManager"> <ref bean="transactionManager"/> <property> </bean>
(2)声明式实现
它是通过AOP框架实现的,具体配置过程:
<bean id="transactionProxyFactoryBean" class="TransactionProxyFactoryBean" <property name="proxyInterfaces"> <list> <value>yourInterfaces</value> </list> </property> <property name="target"...... <property name="transactionManager"..... <property name="transactionAttributeSource".......... </bean> //transactionProxyFactoryBean的命名规范,它继承目标对象的名称,而目标对象的id加Target
事务属性
Spring中事务属性是对事务策略如何应用到方法的描述,这个描述包括下列参数:
传播行为
定义了关于客户端和被调用方法的事务边界,Spring中有7种不同的值
(1)propagation_mandatory方法必须在事务中,没有事务则异常 (2)propagation_nested如果有事务则嵌套,没有则和required相同 (3)propagation_never不在事务中,若有事务则异常 (4)propagation_not_supported方法不在事务中,若有事务,方法运行期间事务挂起 (5)propagation_required必须在事务中,如果没有创建新的事务 (6)propagation_requires_new必运行在自己的事务中,若有事务则方法运行期间挂起 (7)propagation_supports不需要事务,但如果有事务,也可运行在事务中
隔离级别
多个事务同时访问一个数据,就会造成脏读dirty read,不可重复读nonrepeatedable read,幻读phantom read
Spring有5种隔离级别
(1)isolation_default使用数据的隔离级别 (2)isolation_read_uncommited运行脏读,幻读和不可重复读 (3)isolation_read_commited运行并发事务提交后读取,避免脏读 (4)isolation_repeatable_read对相同数据多次读取结果一致,防止脏读,不可重复读 (5)isolation_serializable完全服从ACID隔离级别,完全锁定在事务中涉及的所有表
只读提示
如果一个事务只对数据库执行读操作,可以声明这个事务为只读,数据库就可以应用一些优化措施。
只读的优化措施是在事务启动时有后端的数据库实施的,所以把只有那些有可能启动新事务的传播行为标示为只读才有意义。
此外,如果使用hibernate作为持久化机制,声明一个只读事务将会把hibernate的flush设置为flush_never,避免和数据库进行不比要得同步,将所有的更新延迟到事务的结束。
事务超时间隔
在那些可能启动新事务的传播行为上设置事务超时间隔才有意义。
声明一个简单的事务策略
TransactionProxyFactoryBean 有一个transactionAttributeSource属性,这个属性被设置成TransactionAttributeSource的实例,TransactionProxyFactoryBean 是作为在方法上查找事务属性的一个参考
TransactionAttributeSource的接口定义
public interface TransactionAttributeSource{ public TransactionAttribute getTransactionAttribut(java.lang.reflect.Method method,java.lang.Class targetClass); }
随着transactionAttributeSource的bean对象的声明,所有被TransactionProxyFactoryBean的目标类代理的方法都被执行到一个事务环境中了,但没有指定哪个方法是事务性的,这个由MatchAlwaysTransactionAttributeSource来指定,它是TransactionAttributeSource的一个最简单的实现,每次调用,都是返回相同的TransactionAttribute [required和isolation_default]
可以通过配置一个DefaultTransactionAttribute的bean,来修改MatchAlwaysTransactionAttributeSource默认的事务属性
(1)配置一个DefaultTransactionAttribute
<bean id="myDefautAttribute" class="****.DefaultTransactionAttribute"> //传播行为 <property name="propagationBehaviorName">/\ <value>propagation_never</value> </property> //隔离级别 <property name="isolationLevleName"> <value>isolation_default</value> </property> </bean>
(2)把myDefautAttribute注入到MatchAlwaysTransactionAttributeSource的transactionAttribute属性中即可
对目标对象的每个方法实施不同的事务管理
使用NameMatchTransactionAttributeSource
NameMatchTransactionAttributeSource可以让你在方法上配置不同的事务,配置一个它的实例bean,并注入到TransactionProxyFactoryBean中的transactionAttributeSource中
NameMatchTransactionAttributeSource实例配置
<bean id="transactionAttributeSource" class="NameMatchTransactionAttributeSource"> <property name="properties"> <props> //指定使用事务的方法名,可以使用通配符 <prop key="Your Method Name"> //传播行为,隔离级别,只读设定,-触发回滚,+异常也提交事务 PROPAGATION,ISOLATION,readonly,-Exceptions,+Exceptions </prop> </props> </property> </bean>
不使用NameMatchTransactionAttributeSource
TransactionProxyFactoryBean中有一个transactionProperties属性,我们可以通过配置这个属性来完成不同方法不同事务的配置
...... <property name="transactionProperties"> <props> //指定使用事务的方法名,可以使用通配符 <prop key="Your Method Name"> //传播行为,隔离级别,只读设定,-触发回滚,+异常也提交事务 PROPAGATION,ISOLATION,readonly,-Exceptions,+Exceptions </prop> </props> </property> ......
用元数据声明事务 page 193
bean的配置
我们通过修改transactionAttributeSource的注入替换为注入AttributesTransactionAtrributeSource即可,下面是AttributesTransactionAtrributeSource的配置
<bean id="transactionAttributeSource" class="AttributesTransactionAtrributeSource"> <contructor-arg> <ref bean="attributesImpl"//见下 <contructor-arg> </bean> //attributesImpl 对象将被事务属性源使用,和底层的元数据实现配合,这样就与JSR-175或者Jakarta-Commons Attributes解耦合 <bean id="attributesImpl" class="org.springframw...CommonsAttributes"> .... </bean>
java方法上加上事务标签
在类或者方法前面的注释中增加doclet标签,标签有以下几种形式
/** *@@attribute-name("arg1",property="arg2") *attribute-name:属性名称;arg1构造方法的参数,property="arg2":设置一个属 性的参数 */ [b]例如给 方法simpleMethod()增加事务描述[/b] /** *@@org.springframework.transaction.interceptor.DefaultTransactionAttribute(propagationBehaviorName="PROPAGATION_REQUIRED")*/ public void simpleMethod(){ ...... } //DefaultTransactionAttribute必须用全限定名,避免IDE删除导入的包 //如果把注释写在class类上,那么事务属性应用于所有的method
修剪事务声明
繁复的XML配置文件
当你选择了TransactionAttributeSource,并且把服务层的方法声明为事务的,并且定义了合适的事务管理器,并完成了如下bean的配置
<bean id="txLoanBS" class="org.springframework.transaction.interceptor.TransactionProxyFactoryBean"> <property name="proxyInterfaces"> <list> <value>cn.ccb.csps.efbuss.bizservice.ITxLoanBS</value> </list> </property> <property name="target"> <ref bean="txLoanBSImplTarget" /> </property> <property name="transactionManager"> <ref bean="transactionManager" /> </property> <property name="transactionAttributeSource"> <ref bean="namingTransactionAttribute" /> </property> </bean> <bean id="txLoanBSImplTarget" class="cn.ccb.csps.efbuss.bizservice.impl.TxLoanBSImpl">
浏览这个配置对象,你会发现好几对服务/目标,你会发现好几个对象的声明,它们的名字暗示它们是服务对象比如txLoanBS,但是它只是TransactionProxyFactoryBean的一个实例,真正的服务对象是被命名为带target后缀并被注入到TransactionProxyFactoryBean中target属性中的bean
一般项目中服务对象都用了同样的方法定义,并设定了相同的事务管理器相同的事务属性源,看上去是一大堆繁复的XML文件
消除繁复的XML配置
Spring提供了2种方法:
(1)bean继承,从父TransactionProxyFactoryBean继承,使用Spring对父bean的支持,即bean的parent属性继承父bean的属性
对于上面的bean的具体实现过程如下
首先通过lazy-init=true,它告诉容器不要创建bean,构造一个抽象的TransactionProxyFactoryBean Bean
<bean id="abstractTxDefinition" class="org.springframework.transaction.interceptor.TransactionProxyFactoryBean" lazy-init="init"> <property name="transactionManager"> <ref bean="transactionManager" /> </property> <property name="transactionAttributeSource"> <ref bean="namingTransactionAttribute" /> </property> </bean>
然后再配置某个具体的事务服务
<bean id="txLoanBS" parent="abstractTxDefinition">
<property name="proxyInterfaces">
<list>
<value>cn.ccb.csps.efbuss.bizservice.ITxLoanBS</value>
</list>
</property>
<property name="target">
<bean id="txLoanBSImplTarget"
class="cn.ccb.csps.efbuss.bizservice.impl.TxLoanBSImpl">
</property>
</bean>
(2)AOP自动代理
首先声明一个bean,成为DefaultAdvisorAutoProxyCreator的实列
<bean id="autoProxy" class="********.DefaultAdvisorAutoProxyCreator"/>
DefaultAdvisorAutoProxyCreator将在context中遍历advisor(对于事务为TransactionAttributeSourceAdvisor)自动用它们来代理匹配advisor的pointcut的所有bean
//事务切面 <bean id="transactionAdvisor" class=""TransactionAttributeSourceAdvisor> <constructor-org> <ref bean="transactionInterceptor"/> </constructor-org> </bean> //事务通知 <bean id="transactionInterceptor" class="TransactionInterceptor"> <property name="transactionManager"> <ref bean="transactionManager"/> </property> <property name="transactionAttributeSource"> <ref bean="transactionAttributeSource"/> </property> </bean> //定义属性源 如果使用NameMatchTransactionAttributeSource配置属性源,比如get*配置为执行事务,那么所有的bean的get方法都会被加上事务,这可能不是我们想要的,因此对于自动代理,我们更好的选择是MethodMapTransactionAttributeSource,它需要指定需要事务化的完整类名和方法名 <bean id="transactionAttributeSource" class="MethodMapTransactionAttributeSource"> <property name="methodMap"> <map> <entry key="cn.ccb..TxLoanBSImpl.get*"> <value>PROPAGATION_REQUIRED</value> </entry> </map> </property> </bean> 接下来我们删除所有的TransactionProxyFactoryBean实例,并把服务层的bean的名称去掉Target就可以了
当用自动代理的时候,事务属性源采用AttributesTransactionAtrributeSource是最好的选择了,使方法事务化和非事务化,只不过是往方法中添加合适的元数据就可以了
发表评论
-
Spring2.0 IOC 新特性
2008-04-10 20:57 1573bean的配置定义 引入idref <pr ... -
Spring 基于acegi的安全控制体系
2008-04-09 16:27 846pages 349 -
Spring RPC
2008-04-09 15:34 2390Spring 连接RMI 服务 <bean id=&q ... -
Spring ORM的整合
2008-04-08 19:24 5567Spring对ORM框架做了整合 Spring负责以下几部分功 ... -
Spring AOP 使用ProxyFactroyBean
2008-04-08 16:18 5ProxyFactroyBean的属性: target:代理 ... -
Spring AOP 定义切入点
2008-04-08 11:15 5342首先我们编写了通知advice,但是我们还不能表达在应用系统的 ... -
Spring 实现 AOP
2008-04-08 00:20 2304用java编写spring通知 spri ... -
Spring AOP 面向切面编程 名词解释
2008-04-07 21:02 2239某些功能在应用系统中有可能经常使用,但是却不能通过继承的方式来 ... -
Spring 监听事件
2008-04-07 20:42 2451在应用系统生命周期中,ApplicationContext会发 ... -
Spring 解析文本信息
2008-04-07 20:32 1186Spring 解析资源文件 我们把资源信息放置到proper ... -
Spring Bean 的后处理
2008-04-07 20:04 1274Bean的后处理 Spring提供了2次机会让你切入到Bea ...
相关推荐
Spring事务管理的目的是确保数据的一致性和完整性,尤其是在多操作、多资源的环境中。本Demo将深入探讨Spring如何实现事务的管理。 首先,Spring提供了两种主要的事务管理方式:编程式事务管理和声明式事务管理。 ...
本资源包提供了进行Spring事务管理开发所需的所有关键库,包括框架基础、核心组件、AOP(面向切面编程)支持、日志处理、编译工具以及与数据库交互的相关jar包。下面将对这些知识点进行详细解释: 1. **Spring框架*...
### Spring事务管理详解 #### 一、Spring事务管理概述 Spring框架提供了强大的事务管理功能,使得开发者能够更方便地管理应用程序中的事务。Spring事务管理主要包括两种类型:编程式事务管理和声明式事务管理。 -...
Spring 框架是Java开发中...理解并熟练掌握Spring事务管理,对于提升应用程序的稳定性和可靠性至关重要。在实际开发中,结合声明式事务管理、事务传播行为、隔离级别和回滚规则,可以有效地确保数据的完整性和一致性。
Spring事务管理.pdf 1.资料 2.本地事务与分布式事务 3.编程式模型 4.宣告式模型
Synchronized锁在Spring事务管理下,导致线程不安全。
本篇将深入探讨Spring事务管理的核心概念、工作原理以及如何使用`spring-tx-3.2.0.RELEASE.jar`这个jar包。 首先,我们需要理解什么是事务。在数据库系统中,事务是一组操作,这些操作被视为一个整体,要么全部完成...
标题“Spring事务管理失效原因汇总”指出了本文的核心内容是分析在使用Spring框架进行事务管理时可能遇到的问题及其原因。描述部分进一步说明了事务失效的后果往往不明显,容易在测试环节被忽略,但在生产环境中出现...
本篇文章将深入探讨Spring事务管理的五种方法,旨在帮助开发者更好地理解和运用这一核心特性。 首先,我们来了解什么是事务。在数据库操作中,事务是一组逻辑操作,这些操作要么全部成功,要么全部失败,确保数据的...
本文将详细介绍Spring事务管理的四种方式:编程式事务管理、声明式事务管理、PlatformTransactionManager接口以及TransactionTemplate。 1. **编程式事务管理**:这是一种手动控制事务的方式,通过在代码中调用`...
### Spring事务管理详解 #### 一、Spring事务管理的重要性及必要性 在现代软件开发中,事务管理是一项至关重要的技术,特别是在涉及数据库操作时。事务能够确保一系列操作要么全部成功,要么全部失败,这对于保持...
Spring事务管理是Spring框架的核心特性之一,它提供了一种强大且灵活的方式来管理应用程序中的事务边界。在企业级Java应用中,事务处理是确保数据一致性、完整性和可靠性的关键部分。本篇文章将深入探讨Spring的事务...
总的来说,Spring事务管理提供了一种灵活、强大的方式来处理应用程序中的事务,无论是在简单还是复杂的事务场景下,都能有效保证数据的一致性和完整性。通过声明式事务管理,开发者可以将关注点从事务细节中解脱出来...
Spring事务管理是Spring框架的核心特性之一,它提供了一种在Java应用中管理和协调数据库事务的标准方式。对于有Java基础的开发者来说,理解并掌握Spring事务管理至关重要,因为这有助于确保数据的一致性和完整性,...
当出现像描述中那样的问题——SQL语句执行出错但事务未回滚时,我们需要深入理解Spring事务管理的配置和机制。以下是一些关键知识点: 1. **Spring事务管理类型**: - **编程式事务管理**:通过`...
spring事务管理几种方式代码实例:涉及编程式事务,声明式事务之拦截器代理方式、AOP切面通知方式、AspectJ注解方式,通过不同方式实例代码展现,总结spring事务管理的一般规律,从宏观上加深理解spring事务管理特性...
在思维导图"Spring Transaction.twd"中,可能包含了Spring事务管理的各个概念和它们之间的关系,如事务的ACID属性(原子性、一致性、隔离性和持久性),事务管理器,以及声明式和编程式事务管理的实现方式。...
### Spring事务管理的方法 #### 一、引言 在企业级应用开发中,事务管理是一项至关重要的技术。Spring框架作为Java领域中一个非常流行的轻量级框架,为开发者提供了多种方式来实现事务管理,其中主要分为编程式...