@transitional事务传播行为
在service类前加上@Transactional,声明这个service所有方法需要事务管理。每一个业务方法开始时都会打开一个事务。
spring默认情况下会对运行期例外(RunTimeException)进行事务回滚。这个例外是unchecked ;如果遇到checked(用户例外)就不回滚。
改变默认规则:
1、让checked例外也回滚:在整个方法前加上 @Transactional(rollbackFor=Exception.class)
@Transactional(rollbackFor=Exception.class) public void delete(Integer id) throws Exception{ String sql = "delete from user where id=?"; jdbcTemplate.update(sql, new Object[] { id }, new int[] {java.sql.Types.INTEGER }); throw new Exception("运行时错误!"); }
Exception.class对应抛出异常的异常类
2、 让unchecked例外不回滚: @Transactional(notRollbackFor=RunTimeException.class)
注意:如果异常被try{}catch{}了,事务就不回滚了,如果想让事务回滚必须再往外抛try{}catch{throw Exception}。
Spring中事务的定义
1、propagation:key属性确定代理应该给哪个方法增加事务行为。这样的属性最重要的部份是传播行为。
有以下选项可供使用:
- Propagation.REQUIRED——业务方法需要在一个事务中运行,如果方法运行时,已经处在一个事务中,那么加入到该事务,否则为自己创建一个新的事务。这是默认的选项。
- Propagation.SUPPORTS——如果业务方法在某个事务范围内被调用,则方法成为该事务的一部分,如果业务方法在事务范围外被调用,则方法在没有事务的环境下执行。
- Propagation.MANDATORY——该属性指定业务方法只能在一个已经存在的事务中执行,业务方法不能发起自己的事务。如果业务方法在没有事务的环境下调用,容器就会抛出例外。
- Propagation.REQUIRES_NEW——不管是否存在事务,业务方法总会为自己发起一个新的事务。如果方法已经运行在一个事务中,则原有事务会被挂起,新的事务会被创建,直到方法执行结束,新建事务才算结束,原先的事务才会恢复执行。
- Propagation.NOT_SUPPORTED——声明方法不需要事务。如果方法没有关联到一个事务,容器不会为它开启事务。如果方法在一个事务中被调用,该事务会被挂起,在方法调用结束后,原先的事务便会恢复执行。
- Propagation.NEVER——指定业务方法绝对不能在事务范围内执行。如果业务方法在某个事务中执行,容器会抛出例外,只有业务方法没有关联到任何事务,才能正常执行。
- Propagation.NESTED——如果活动的事务存在,则运行在一个嵌套的事务中,如果没有活动事务,则按照Propagation.REQUIRED属性执行。它使用了一个单独的事务,这个事务拥有多个可以回滚的保存点,内部事务的回滚不会对外部事务造成影响。它只对DataSourceTransactionManager事务管理器起效。
实例说明:
ServiceA{ @Transactional(propagation=Propagation.REQUIRED) void methodA() { ServiceB.methodB(); } } ServiceB{ @Transactional(propagation=Propagation.REQUIRED) void methodB() { } }
- 1:Propagation.REQUIRED
ServiceB.methodB的事务级别定义为Propagation_REQUIRED, 那么由于执行ServiceA.methodA的时候, ServiceA.methodA已经起了事务,这时调用ServiceB.methodB,ServiceB.methodB看到自己已经运行在ServiceA.methodA的事务内部,就不再起新的事务。而假如ServiceA.methodA运行的时候发现自己没有在事务中,他就会为自己分配一个事务。这样,在ServiceA.methodA或者在ServiceB.methodB内的任何地方出现异常,事务都会被回滚。即使ServiceB.methodB的事务已经被提交,但是ServiceA.methodA在接下来fail要回滚,ServiceB.methodB也要回滚。 - 2:Propagation.SUPPORTS
如果ServiceB.methodB当前在事务中,即以事务的形式运行;如果ServiceB.methodB当前不在一个事务中,那么就以非事务的形式运行。这就跟平常用的普通非事务的代码只有一点点区别了。 - 3:Propagation_MANDATORY
必须在一个已经存在的事务中运行。也就是说,他只能被一个父事务调用。否则,他就要抛出异常。 - 4:Propagation_REQUIRES_NEW
比如我们设计ServiceA.methodA的事务级别为Propagation_REQUIRED,ServiceB.methodB的事务级别为Propagation_REQUIRES_NEW,那么当执行到ServiceB.methodB的时候,ServiceA.methodA所在的事务就会挂起,ServiceB.methodB会起一个新的事务,等待ServiceB.methodB的事务完成以后,它才继续执行。它与Propagation_REQUIRED 的事务区别在于事务的回滚程度了。因为ServiceB.methodB是新起一个事务,那么就是存在两个不同的事务。如果ServiceB.methodB已经提交,那么ServiceA.methodA失败回滚,ServiceB.methodB是不会回滚的。如果ServiceB.methodB失败回滚,如果它抛出的异常被ServiceA.methodA捕获,ServiceA.methodA事务仍然可能提交。 - 5:Propagation.NOT_SUPPORTED
当前不支持事务。比如ServiceA.methodA的事务级别是Propagation_REQUIRED ,而ServiceB.methodB的事务级别是Propagation_NOT_SUPPORTED ,那么当执行到ServiceB.methodB时,ServiceA.methodA的事务挂起,而他以非事务的状态运行完,再继续ServiceA.methodA的事务。 - 6:Propagation_NEVER
不能在事务中运行。假设ServiceA.methodA的事务级别是Propagation_REQUIRED, 而ServiceB.methodB的事务级别是Propagation_NEVER ,那么ServiceB.methodB就要抛出异常了。 - 7:Propagation_NESTED
ServiceA { @Transactional(propagation=Propagation.REQUIRED) void methodA() { Connction conn=null; System.out.println("我是A方法"); try { ServiceB.methodB();//Propagation_NESTED 级别 } catch (Exception) { conn.rollback(); } } } ServiceB{ @Transactional(propagation=Propagation_NESTED) void methodB() { System.out.println("我是B方法"); } } //ServiceA.methodA调用ServiceB.methodB内部运行过程 System.out.println("我是A方法"); //savepoint Savepoint savepoint=conn.setSavepoint(); try{ System.out.println("我是B方法"); }catch(Excption){ conn.rollback(savepoint); }
理解Nested的关键是savepoint。他与Propagation_REQUIRES_NEW的区别是,Propagation_REQUIRES_NEW另起一个事务,将会与他的父事务相互独立,而Nested的事务和他的父事务是相依的,他的提交是要等和他的父事务一块提交的。当运行到ServiceB.methodB方法时,检测到其事务类型为Propagation_NESTED ,spring容器会自动为其建一个保存点(savepoint)当ServiceB.methodB失败,则回滚到保存点,不影响ServiceA.methodA的执行。假如ServiceA.methodA失败回滚,即使ServiceB.methodB运行成功,整个事务也要回滚。
2、readOnly
事务属性中的readOnly标志表示对应的事务应该被最优化为只读事务。这是一个最优化提示。在一些情况下,一些事务策略能够起到显著的最优化效果,例如在使用Object/Relational映射工具(如:Hibernate或TopLink)时避免dirty checking(试图“刷新”)。在做查询时很有用。
3、Timeout
在事务属性中还有定义“timeout”值的选项,指定事务超时为几秒。默认为30秒,一般不改这个属性。在JTA中,这将被简单地传递到J2EE服务器的事务协调程序,并据此得到相应的解释。
数据库事务隔离级别
隔离级别在于处理多事务的并发问题。数据库系统提供了四种事务隔离级别:
- Serializable:事务串行执行。最严格的级别,资源消耗最大。
- Repeatable Read:避免了“脏读”和“不可重复读取”的情况,但会出现”幻读“。
- Read Commited:避免了“脏读”,但会出现”幻读“、”不可重复读取“。大多数主流数据库的默认事务等级。
- Read Uncommitted:避免读取过程中读取到非法数据,但会出现“脏读”、”幻读“、”不可重复读取“。
注:
- 脏读:一个事务读取到另一个事务未提交的更新数据。
- 不可重复读:在同一个事务中,多次读取同一数据返回的结果有所不同。即,后续读取的数据是另一个事务已提交的更新数据。
- 幻读:一个事务读取到另一个事务已提交的insert数据。
发表评论
-
spring+jdbc组合开发
2011-07-31 10:04 8071、架包:commons-dbcp.jar commons- ... -
spring+jdbc组合开发
2011-07-28 23:13 7spring+jdbc组合开发 1、架包:commons-d ... -
AOP_使用spring框架进行面向切面编程
2011-07-25 21:33 1181用spring实现AOP有两种方 ... -
AOP_基本概念
2011-07-23 00:00 707分散关注 将通用需求功能从不相关类之中分离出来;同时,能 ... -
AOP_使用CGLIB实现AOP功能
2011-07-22 00:02 784好处:不需要目标对象实现接口 使用的架包:cglib-nod ... -
AOP__使用JDK中的Proxy技术实现AOP功能
2011-07-21 22:28 539AOP主要用于权限的验证 细粒度的权限控制:对方法进行拦截, ... -
DI依赖注入_自动装配_autowire
2011-07-20 21:27 695自动装入:容器自动装 ... -
DI依赖注入_手动装配_注解基础
2011-07-20 22:50 1306以下的注释不管在那种配置组件的方法中都能用,这里只是实际需要才 ... -
DI依赖注入_手动装配_非注解
2011-07-18 23:15 915基本数据类型的注入(将以下代码放到对象的bean中即可) 可 ... -
spring中的方法
2011-07-18 08:03 01、实例化spring容器 ApplicationConte ... -
spring基本概念
2011-07-17 21:38 575Ioc(控制反转):应用本 ...
相关推荐
在 web 开发中,安全性是一个非常值得我们关注的问题,涉及到两个方面:程序安全性和数据库安全性。数据库安全性是我们需要关注的关键问题之一,备份数据库可以帮助我们防止数据丢失和灾难恢复。 备份数据库 备份...
3. **编写一个简单的显示数据的网页:** 最终目标是使用 ASP.NET 和数据库来实现一个可以展示数据的简单网页应用。 #### 二、SQL Server 操作练习 ##### 1. 创建数据库和表 - **创建 test 数据库:** - 使用 SQL...
在本案例中,我们将讨论如何利用Lucene.NET对数据库中的数据进行索引和搜索。 ### 1. 数据库表结构 在建立索引前,我们需要了解数据库的表结构。例如,这里有一个名为`userblog`的博客表,包含以下字段: - `id`...
《XHTML1-transitional.dtd 深入学习》 在网页设计和开发中,DTD(Document Type Definition)扮演着...通过不断学习和实践,我们可以更好地理解和掌握XHTML1-transitional.dtd,提升网页的可维护性和跨浏览器兼容性。
根据提供的信息,我们可以了解到这篇文档的主要目标是介绍如何在 SQL Server 中不借助任何第三方工具来生成数据库字典。下面将详细解析并展开该主题相关的知识点。 ### 一、背景及目的 在数据库管理中,数据库字典...
在JSP(JavaServer Pages)开发中,数据库连接池的配置是一项非常重要的技术,它能够显著提高应用程序的性能和响应速度。本文将详细介绍如何在JSP项目中配置MySQL数据库连接池。 #### 一、准备环境与所需文件 1. *...
根据提供的文件信息,我们可以分析出该段代码是用于通过ODBC(Open Database Connectivity,开放数据库互连)方式访问数据库,并展示部分数据记录的Java Web应用示例。下面将对该示例进行详细的解析。 ### ODBC简介...
在IT领域,多文件上传至数据库是一个常见的需求,特别是在web应用程序中,如内容管理系统、社交媒体平台等。这里我们将深入探讨如何实现这个功能,主要涉及Java技术栈,特别是使用Struts2框架。 首先,我们需要创建...
- 这里没有详细说明 JDTS 的安装步骤,但通常指的是 Java Database Tools Set 或者其他用于管理和操作数据库的工具集。 ##### 4. Tomcat 配置文件修改 - 修改 `%CATALINA_HOME%\conf\web.xml` 文件中的 `...
日志记录是数据库恢复中的另一个核心机制,主要用于记录事务处理过程中的所有更改操作。通过分析这些日志记录,可以在故障后重建数据库状态。 #### 3. 影子拷贝 影子拷贝技术通常用于创建数据库的实时副本,这有助...
### 在Tomcat 7.0中配置数据库连接池 #### 一、背景介绍 在现代Web应用程序开发过程中,数据库操作是非常频繁的一项任务。为了提高数据库访问效率并减少资源开销,通常会采用数据库连接池技术。Apache Tomcat作为一...
对于数据库操作而言,我们通常会利用JDBC(Java Database Connectivity)技术来实现与数据库之间的连接和交互。 #### 二、准备工作 在进行数据库操作前,我们需要做一些准备工作: 1. **配置JDBC驱动**:确保项目...
ASP+MySQL数据库连接配置详解 本文详细介绍了ASP+MySQL数据库连接配置的相关知识点,涵盖了配置ODBC数据源、不同的数据验证方式、...通过本文的学习,可以帮助读者更好地理解ASP+MySQL数据库连接配置的原理和实现。
### 使用SQL查询生成SQL Server数据库结构字典并输出为HTML ...这种方式不仅提高了工作效率,还便于非技术人员理解和使用数据库结构信息。同时,根据实际需求调整查询脚本和HTML模板,还可以实现更多的定制化功能。
通过上述内容,我们可以了解到如何利用JDBC-ODBC桥接器实现Java程序与Microsoft Access数据库之间的交互,包括如何设置环境、编写表单和Servlet,以及解决可能出现的问题。这对于初学者来说是非常宝贵的实践经验和...
在本文中,我们将探讨如何使用JavaServer Pages (JSP)技术连接到Microsoft Access数据库。这是一个常见的任务,特别...在实际应用中,还需要考虑异常处理、事务管理、连接池等高级主题,以提高应用程序的稳定性和性能。
3. 数据库管理:虽然Access数据库简单易用,但其性能和扩展性有限,对于大型项目,建议使用更强大的数据库系统,如MySQL或Oracle。 4. 部署环境:由于Access数据库的文件格式,当在多用户环境中部署时,可能需要...
URL格式指定了数据库的类型、服务器地址、端口号和数据库名称。 - `createStatement()`方法创建一个`Statement`对象,用于执行SQL查询。这里使用了`TYPE_SCROLL_INSENSITIVE`和`CONCUR_READ_ONLY`参数,表示结果集...
DOCTYPE HTML PUBLIC "-//W3C//DTD HTML 4.01 Transitional//EN"> <base href="<%=basePath%>"> 用户注册 <meta http-equiv="expires" content="0"> ,keyword2,keyword3"> <!-- --> ...