Java 提供了两类主要的异常 :runtime exception 和 checked exception 。 checked 异常也就是我们经常遇到的 IO 异常,以及 SQL 异常都是这种异常。 对于这种异常, JAVA 编译器强制要求我们必需对出现的这些异常进行 catch 。所以,面对这种异常不管我们是否愿意,只能自己去写一大堆 catch 块去处理可能的异常。
但是另外一种异常: runtime exception ,也称运行时异常,我们可以不处理。当出现这样的异常时,总是由虚拟机 接管。比如:我们从来没有人去处理过 NullPointerException 异常,它就是运行时异常,并且这种异常还是最常见的异常之一。
出现运行时异常后,系统会把异常一直往上层抛,一直遇到处理代码。如果没有处理块,到最上层,如果是多线程就由 Thread.run() 抛出 ,如果是单线程就被 main() 抛出 。抛出之后,如果是线程,这个线程也就退出了。如果是主程序抛出的异常,那么这整个程序也就退出了。运行时异常是 Exception 的子类,也有一般异常的特点,是可以被 Catch 块处理的。只不过往往我们不对他处理罢了。也就是说,你如果不对运行时异常进行处理,那么出现运行时异常之后,要么是线程中止,要么是主程序终止。
如果不想终止,则必须扑捉所有的运行时异常,决不让这个处理线程退出。队列里面出现异常数据了,正常的处理应该是把异常数据舍弃,然后记录日志。不应该由于异常数据而影响下面对正常数据的处理。 在这个场景这样处理可能是一个比较好的应用,但并不代表在所有的场景你都应该如此。如果在其它场景,遇到了一些错误,如果退出程序比较好,这时你就可以不太理会运行时异常 ,或者是通过对异常的处理显式的控制程序退出。
异常处理的目标之一就是为了把程序从异常中恢复出来 。
REQUIRED--支持当前事务,如果当前没有事务,就新建一个事务。这是最常见的选择。
SUPPORTS--支持当前事务,如果当前没有事务,就以非事务方式执行。
MANDATORY--支持当前事务,如果当前没有事务,就抛出异常。
REQUIRES_NEW--新建事务,如果当前存在事务,把当前事务挂起。
NOT_SUPPORTED--以非事务方式执行操作,如果当前存在事务,就把当前事务挂起。
NEVER--以非事务方式执行,如果当前存在事务,则抛出异常。
NESTED--嵌套事物。
srping增加了NESTED,其它类型与EJB是一样的。
需要明确一点,方法上指定的事物传播属性都是指被另一个service调用的传播属性。对于同一个service内的方法间调用则始终可以看作是REQUIRED,即在同一个事物中。因此即使方法A指定的是NEVER,并且它还调用了内部另一个方法B,传播属性是MANDATORY,这时个同样能正常执行。
为说明问题,假设有两个service,分别是S1和S2
事物的回滚与否是按方法最终抛出的异常来评定,即以方法为单位,方法过程中抛出什么异常不管,只要你捕获了就可以保证事物不回滚。但值得注意的是,在 service调service时,比如S1中的方法A调用了S2中的方法B,如果B抛出了unchecked异常,并且A和B在同一个事物中,即使A中捕获了B的异常,A的事物同样会回滚。
所以讨论事物传播需要区分service间的调用和service内部调用。
1:service间的调用
在同一个事物里,spring容器会在方法执行完毕后检查方法是否有抛出引发回滚的异常(unchecked)。当遇到unchecked异常时,该事物会回滚,即使将异常捕获了,在方法执行最终还是会回滚(相当于最终通知来做判断事物是否回滚),。如果遇到的是checked异常,即使不捕获,在异常前发生的数据操作一样生效。
引发回滚的异常来源有两个:1.该service方法内本身抛出unchecked异常.2.调用的其它service方法抛出unchecked异常。即使在调用的service方法中被捕获了异常,虽然不提倡这样做!如果两个方法在同一个事物里,那么该事物还是会回滚,因为这时候它会告诉spring容器当前事物需要回滚,在该事物最后spring容器会根据信号来判断是否回滚。
2:service内部调用
service内部的方法间调用,比如A方法里调用了B方法,那么B方法的事物传播属性就将被忽略。需要注意的是它实际上是将本service内的其它方法看作是一个代码段(就是在一个事物下,不存在事物传播问题),就是说如果B方法抛出了异常,哪怕是unchecked异常,只要在A方法内被捕获了,就相当于异常可处理并被处理了,因此事物不回滚。
因此要紧记住一点unchecked表示未知且不应该被处理的异常,即使被处理了,spring容器也会被告知事物回滚,对于checked异常,spring容器是不管的,即使抛出去了,事物照样不回滚。因此在设计方法时一定要遵守这样一个原则,不要捕获unckecked异常(或者捕获后再抛出),对于checked异常如果不能处理也以uncheck异常抛出。因此service应该是一个只抛unchecked异常。
tip:如果一个标记事物标签的方法调用了一个未标记的方法,则同样会被加入到事物管理里面。当然如上所述的前题都必需在同一个事物管理器里面。
明明在spring的配置文件里设置好了事务
但他出错后偏偏就是不回滚
现在我需要达到的效果是更新数据库中的多个表
分两步骤完成
1. 删除A 、B、 C表
2. 向A、B、C表添加新数据
若第二步出错,则需要回滚,撤销删除动作。
最初为了抛出异常我在删除C表的时候故意操作一个不存在表,但结果并没有回滚A 、B表被无情删除;
网上查了下原因,原来是mysql表的属性设置引起的。
默认下mysql表的stroage engine 属性是“MyISAM ”不支持事务,要修改为InnoDB。
修改后表属性后,在 删除C表出错时,删除A、B的操作会被回滚。
继续测试发现,在删除A 、B、 C表后,向A、B、新数据后立即通过 throw new Exception("")抛出一个异常,结果前面的动作会照常提交到数据库,没有回滚; 但如果我以运行int n=1/0的方式制造一个异常,则前面的删除动作就可以回滚了。
非常纳闷,这究竟是什么原因呢?
以下是spring中关于事务的配置
- < import resource = "dao-jdbc-config.xml" />
- < bean id = "transactionManager"
- class ="org.springframework.jdbc.datasource.DataSourceTransactionManager" >
- < property name = "dataSource" >
- < ref bean = "dataSource" />
- </ property >
- </ bean >
- < bean id = "transactionInterceptor"
- class = "org.springframework.transaction.interceptor.TransactionInterceptor" >
- < property name = "transactionManager" ref = "transactionManager" />
- < property name = "transactionAttributes" >
- < props >
- < prop key = "*" > PROPAGATION_REQUIRED </ prop >
- < prop key = "add*" > PROPAGATION_REQUIRED </ prop >
- < prop key = "del*" > PROPAGATION_REQUIRED </ prop >
- < prop key = "update*" > PROPAGATION_REQUIRED </ prop >
- < prop key = "select*" > PROPAGATION_REQUIRED,readOnly </ prop >
- < prop key = "get*" > PROPAGATION_REQUIRED,readOnly </ prop >
- </ props >
- </ property >
- </ bean >
- < bean
- class ="org.springframework.aop.framework.autoproxy.BeanNameAutoProxyCreator" >
- < property name = "beanNames" >
- < value > *Service </ value >
- </ property >
- < property name = "interceptorNames" >
- < list >
- < value > transactionInterceptor </ value >
- <!--
- 此处增加新的Interceptor
- -->
- </ list >
- </ property >
- </ bean >
- < bean
- class ="org.springframework.transaction.interceptor.TransactionAttributeSourceAdvisor" >
- < property name = "transactionInterceptor"
- ref = "transactionInterceptor" />
- </ bean >
在添上用svervice方法里面的操作
Checked exception 是在编译时在语法上必须处理的异常,因此必须在语法上以try..catch加以处理;
Unchecked exception是运行时异常,它继承java.lang.RuntimeException
DataAccessException 就属于RuntimeException
相关推荐
很好的spring+ibatis事务的配置文档.
给定的XML配置文件展示了如何在Spring中配置ibatis数据源以及ibatis事务管理器。首先,注释掉的`BasicDataSource`部分是用来配置数据源的,这里没有启用,而是选择了使用JNDI数据源。实际生产环境中,通常会使用...
以下是关于"spring ibatis 配置(包括事务管理)"的详细说明: 1. **引入依赖**:首先,我们需要在项目中添加Spring和iBatis的相关依赖。通常,这会在Maven或Gradle的配置文件中完成,确保引入了正确的版本。 2. *...
根据提供的文件信息,本文将详细解析Spring与iBatis整合时如何保留并使用iBatis事务管理机制,以及如何在应用程序中实现手动控制事务的方法。 ### Spring与iBatis整合 Spring框架是一个全面的企业级应用开发框架,...
本文将深入探讨如何利用Ibatis实现一对多关系、批处理、事务管理和与Spring及Struts2的集成。 首先,让我们来看一下“一对多”关系。在数据库设计中,一对多关系很常见,比如一个用户可以有多个订单。在Ibatis中,...
本示例“spring+ibatis声明式事务Demo”将探讨如何在Spring与iBatis集成环境中使用声明式事务管理。声明式事务管理允许开发者通过配置来控制事务,无需在代码中显式处理事务开始、提交和回滚。 **Spring框架** 是一...
Spring是一个全面的后端开发框架,提供了依赖注入、AOP(面向切面编程)、事务管理等特性,而iBatis则是一个优秀的持久层框架,它将SQL语句与Java代码分离,使得数据库操作更加灵活和易于维护。将两者整合可以实现松...
当我们谈论"spring+ibatis声明式事务Demo"时,我们关注的是如何在Spring框架中利用iBatis实现声明式事务管理。 声明式事务管理是Spring框架提供的一种方便、高效的方式,使得开发者无需手动控制事务的开始、提交、...
同时,Spring的事务管理能力确保了对数据库操作的原子性和一致性,通过声明式事务管理,可以在XML配置文件中轻松设置事务边界,如开启、提交、回滚等操作。 Ibatis是轻量级的持久层框架,它简化了JDBC的繁琐工作,...
本篇将详细介绍如何在Spring 2.5版本中整合iBATIS 2.3,并利用Spring的声明式事务管理,以提升应用程序的稳定性和可维护性。 首先,我们需要了解Spring 2.5和iBATIS 2.3的基本概念。Spring 2.5是Spring框架的一个...
本案例聚焦于`ibatis`(MyBatis前身)的事务管理,它是一个轻量级的Java持久层框架,提供了灵活的SQL映射功能。在Java应用中,Ibatis与SQL Server、MySQL等数据库配合,可以实现高效的数据库访问和事务处理。 **1. ...
Spring提供了IOC(Inversion of Control,控制反转)和AOP(Aspect-Oriented Programming,面向切面编程)等核心特性,使得开发者可以轻松地进行事务管理、安全管理以及与其他框架的集成。在本项目中,Spring可能被...
本例子通过Spring容器管理ibatis dwr2的javaBean以及事务,Extjs通过dwr2来调用Spring管理类
Spring-iBatis会自动处理SQL执行、结果转换以及事务管理。 8. **优点** - **松耦合**:SQL与Java代码分离,降低了维护难度。 - **灵活性**:iBatis允许动态SQL,适应复杂查询需求。 - **事务管理**:Spring的...
Spring 是一个全面的后端开发框架,提供依赖注入、面向切面编程、事务管理等功能;而 iBATIS 是一个持久层框架,它简化了数据库操作,将 SQL 语句与 Java 代码分离,提供了灵活的数据访问接口。 在整合 Spring 和 ...
它允许开发者轻松管理对象间的依赖关系,同时提供了事务管理、安全控制等企业级服务。 3. **Hibernate**:Hibernate是一个强大的ORM(对象关系映射)框架,用于简化数据库操作。它允许开发者用Java对象来表示数据库...
同时,Spring的AOP(面向切面编程)功能可用于事务管理,确保数据操作的一致性。 iBatis作为轻量级的持久层框架,它提供了一种SQL映射机制,将SQL语句与Java代码分离,提高了代码的可读性和可维护性。在本项目中,...