Spring事务管理(一)
对于Spring相信很多做web开发的小活动一定不陌生,Spring中我们经常谈到的就是IOC和AOP,但是对于Spring的事务管理,相信大家一定也很感兴趣,今天我们就探讨一下Spring中的事务管理。
首先谈一下事务使用的场景,我们能想到的最常见场景就是银行转账,A给B转账,第一步扣除A中的账户金额,第二步将扣除的金额加入到B账户,这个过程就要求,这两步,必须同时成功,如果失败就需要进行事务回滚,不然我们的数据就会出现异常。这个怎么来进行实现呢?这就要从今天的事务说起了。首先谈一下事务的4大特性:原子性(Atomicity)、一致性(Consistency)、隔离性(Isolation)、持久性(Durability)。
⑴ 原子性(Atomicity)
原子性是指事务包含的所有操作要么全部成功,要么全部失败回滚,这和前面两篇博客介绍事务的功能是一样的概念,因此事务的操作如果成功就必须要完全应用到数据库,如果操作失败则不能对数据库有任何影响。
⑵ 一致性(Consistency)
一致性是指事务必须使数据库从一个一致性状态变换到另一个一致性状态,也就是说一个事务执行之前和执行之后都必须处于一致性状态。
拿转账来说,假设用户A和用户B两者的钱加起来一共是5000,那么不管A和B之间如何转账,转几次账,事务结束后两个用户的钱相加起来应该还得是5000,这就是事务的一致性。
⑶ 隔离性(Isolation)
隔离性是当多个用户并发访问数据库时,比如操作同一张表时,数据库为每一个用户开启的事务,不能被其他事务的操作所干扰,多个并发事务之间要相互隔离。
即要达到这么一种效果:对于任意两个并发的事务T1和T2,在事务T1看来,T2要么在T1开始之前就已经结束,要么在T1结束之后才开始,这样每个事务都感觉不到有其他事务在并发地执行。
关于事务的隔离性数据库提供了多种隔离级别,稍后会介绍到。
⑷ 持久性(Durability)
持久性是指一个事务一旦被提交了,那么对数据库中的数据的改变就是永久性的,即便是在数据库系统遇到故障的情况下也不会丢失提交事务的操作。
例如我们在使用JDBC操作数据库时,在提交事务方法后,提示用户事务操作完成,当我们程序执行完成直到看到提示后,就可以认定事务以及正确提交,即使这时候数据库出现了问题,也必须要将我们的事务完全执行完成,否则就会造成我们看到提示事务处理完毕,但是数据库因为故障而没有执行事务的重大错误。
以上介绍完事务的四大特性(简称ACID),现在重点来说明下事务的隔离性,当多个线程都开启事务操作数据库中的数据时,数据库系统要能进行隔离操作,以保证各个线程获取数据的准确性,在介绍数据库提供的各种隔离级别之前,我们先看看如果不考虑事务的隔离性,会发生的几种问题:1、脏读:脏读是指在一个事务处理过程里读取了另一个未提交的事务中的数据。2、不可重复读:不可重复读是指在对于数据库中的某个数据,一个事务范围内多次查询却返回了不同的数据值,这是由于在查询间隔,被另一个事务修改并提交了。3、虚读(幻读):幻读是事务非独立执行时发生的一种现象。幻读和不可重复读都是读取了另一条已经提交的事务(这点就脏读不同),所不同的是不可重复读查询的都是同一个数据项,而幻读针对的是一批数据整体(比如数据的个数)。
说了这么多,下面我们我们开始通过程序来简单实现一下转账场景中如何使用事务。
首先我们这里需要搭建一个Spring下的转账场景,我使用的是jdk8+maven+spring+jdbc来模拟转账场景:
1、打开eclipse(最新版本的直接支持maven环境),创建我们的maven项目
2、打开pom.xml添加我们的jar依赖:
- <project xmlns="http://maven.apache.org/POM/4.0.0" xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"
- xsi:schemaLocation="http://maven.apache.org/POM/4.0.0 http://maven.apache.org/xsd/maven-4.0.0.xsd">
- <modelVersion>4.0.0</modelVersion>
- <groupId>com.edu.hpugs.spring</groupId>
- <artifactId>springTransation</artifactId>
- <version>0.0.1-SNAPSHOT</version>
- <packaging>jar</packaging>
- <name>springTransation</name>
- <url>http://maven.apache.org</url>
- <properties>
- <project.build.sourceEncoding>UTF-8</project.build.sourceEncoding>
- </properties>
- <dependencies>
- <!-- https://mvnrepository.com/artifact/mysql/mysql-connector-java -->
- <dependency>
- <groupId>mysql</groupId>
- <artifactId>mysql-connector-java</artifactId>
- <version>6.0.6</version>
- </dependency>
- <!-- https://mvnrepository.com/artifact/org.springframework/spring-jdbc -->
- <dependency>
- <groupId>org.springframework</groupId>
- <artifactId>spring-jdbc</artifactId>
- <version>4.3.11.RELEASE</version>
- </dependency>
- <!-- https://mvnrepository.com/artifact/org.springframework/spring -->
- <dependency>
- <groupId>org.springframework</groupId>
- <artifactId>spring</artifactId>
- <version>2.5.6</version>
- </dependency>
- <!-- https://mvnrepository.com/artifact/org.springframework/spring-context -->
- <dependency>
- <groupId>org.springframework</groupId>
- <artifactId>spring-context</artifactId>
- <version>4.3.11.RELEASE</version>
- </dependency>
- <!-- https://mvnrepository.com/artifact/org.springframework/spring-core -->
- <dependency>
- <groupId>org.springframework</groupId>
- <artifactId>spring-core</artifactId>
- <version>4.3.11.RELEASE</version>
- </dependency>
- <!-- https://mvnrepository.com/artifact/org.springframework/spring-beans -->
- <dependency>
- <groupId>org.springframework</groupId>
- <artifactId>spring-beans</artifactId>
- <version>4.3.11.RELEASE</version>
- </dependency>
- <!-- https://mvnrepository.com/artifact/org.springframework/spring-test -->
- <dependency>
- <groupId>org.springframework</groupId>
- <artifactId>spring-test</artifactId>
- <version>4.3.11.RELEASE</version>
- <scope>test</scope>
- </dependency>
- <!-- https://mvnrepository.com/artifact/org.apache.logging.log4j/log4j-core -->
- <dependency>
- <groupId>org.apache.logging.log4j</groupId>
- <artifactId>log4j-core</artifactId>
- <version>2.9.0</version>
- </dependency>
- <dependency>
- <groupId>junit</groupId>
- <artifactId>junit</artifactId>
- <version>4.12</version>
- <scope>test</scope>
- </dependency>
- </dependencies>
- </project>
这里我们使用的是mysql数据库,所以添加mysql驱动jar包;这里我们使用spring-jdbc连接数据,所以引入spring-jdbc包;其次就是我们的spring相关jar包了
3、编写我们的service、dao层:
a、service接口:
- public interface IBlankService {
- void transferAccounts(final String outName, final String inName, final double monery);
- }
b、service实现:
- public class BlankServiceImpl implements IBlankService {
- private IBlankDao blankDao;
- public void setBlankDao(IBlankDao blankDao) {
- this.blankDao = blankDao;
- }
- public void transferAccounts(String outName, String inName, double monery) {
- // TODO Auto-generated method stub
- blankDao.outMonery(outName, monery);
- int i = 1 / 0;
- blankDao.inMonery(inName, monery);
- }
- }
c、dao接口:
- public interface IBlankDao {
- void outMonery(final String userName, final double monery);
- void inMonery(final String userName, final double monery);
- }
d、dao实现:
- public class BlankDaoImpl implements IBlankDao {
- private DataSource dataSource;
- public DataSource getDataSource() {
- return dataSource;
- }
- public void setDataSource(DataSource dataSource) {
- this.dataSource = dataSource;
- }
- public void outMonery(String userName, double monery) {
- // TODO Auto-generated method stub
- String sql = "UPDATE userAccount SET monery = monery - ? WHERE name = ?";
- JdbcTemplate jdbcTemplate = new JdbcTemplate(dataSource);
- jdbcTemplate.update(sql, monery, userName);
- }
- public void inMonery(String userName, double monery) {
- // TODO Auto-generated method stub
- String sql = "UPDATE userAccount SET monery = monery + ? WHERE name = ?";
- JdbcTemplate jdbcTemplate = new JdbcTemplate(dataSource);
- jdbcTemplate.update(sql, monery, userName);
- }
- }
4、这里我们通过junit单元测试的方法来调用我们的service:
- public class UserTest {
- private IBlankService blankService;
- @Before
- public void before(){
- ApplicationContext appletContext = new ClassPathXmlApplicationContext("applicationContext.xml");
- blankService = (IBlankService) appletContext.getBean("blankService");
- }
- @Test
- public void userTest(){
- blankService.transferAccounts("aa", "bb", 100);
- }
- }
5、我们的resource文件:
a、applicationContext.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" xmlns:aop="http://www.springframework.org/schema/aop"
- xmlns:context="http://www.springframework.org/schema/context" xmlns:tx="http://www.springframework.org/schema/tx"
- xsi:schemaLocation="http://www.springframework.org/schema/beans http://www.springframework.org/schema/beans/spring-beans.xsd
- http://www.springframework.org/schema/aop http://www.springframework.org/schema/aop/spring-aop-4.3.xsd
- http://www.springframework.org/schema/context http://www.springframework.org/schema/context/spring-context-4.3.xsd
- http://www.springframework.org/schema/tx http://www.springframework.org/schema/tx/spring-tx-4.3.xsd">
- <context:property-placeholder location="classpath:db.properties"/>
- <bean id="dataSource" class="org.springframework.jdbc.datasource.DriverManagerDataSource">
- <property name="driverClassName" value="${jdbc.dirverClass}" />
- <property name="url" value="${jdbc.url}" />
- <property name="username" value="${jdbc.user}" />
- <property name="password" value="${jdbc.password}" />
- </bean>
- <bean id="blankService" class="com.edu.hpugs.spring.BlankServiceImpl">
- <property name="blankDao" ref="blankDao"></property>
- </bean>
- <bean id="blankDao" class="com.edu.hpugs.spring.BlankDaoImpl">
- <property name="dataSource" ref="dataSource"></property>
- </bean>
- </beans>
b、db.properties
- jdbc.dirverClass=com.mysql.jdbc.Driver
- jdbc.url=jdbc\:mysql\://127.0.0.1\:3306/blank?useUnicode=true&characterEncoding=UTF-8&useJDBCCompliantTimezoneShift=true&useLegacyDatetimeCode=false&serverTimezone=UTC
- jdbc.user=root
- jdbc.password=root
c、log4j.properties
- #log4j.rootCategory=error,stdout
- #
- #log4j.appender.stdout=org.apache.log4j.ConsoleAppender
- #log4j.appender.stdout.layout=org.apache.log4j.PatternLayout
- #log4j.appender.stdout.layout.ConversionPattern=[HYYT] %p [%t] %C.%M(%L) | %m%n
- #
- #log4j.logger.com.opensymphony.xwork2.ognl.OgnlValueStack=ERROR
- ##########################\u914d\u7f6erootLogger--\u8f93\u51fa\u6d88\u606f\u7ea7\u522b\uff1aINFO, WARN, ERROR\u548c FATAL ###############
- #log4j.rootLogger = ALL, STUDIO, INFO_FILE, WARN_FILE, ERROR_FILE, FATAL_FILE
- #####################################\u63a7\u5236\u53f0\u8f93\u51fa##################################################
- log4j.rootLogger = ALL, INFO_FILE, WARN_FILE, ERROR_FILE
- ###############################################################################################
- log4j.appender.STUDIO = org.apache.log4j.ConsoleAppender
- log4j.appender.STUDIO.Targer = System.out
- log4j.appender.STUDIO.Threshold = INFO
- log4j.appender.STUDIO.ImmediateFlush = TRUE
- log4j.appender.STUDIO.layout = org.apache.log4j.PatternLayout
- log4j.appender.STUDIO.layout.ConversionPattern = [%-5p][%-22d{yyyy/MM/dd HH\:mm\:ss}][%l]%n%m%n
- ####################################INFO_FILE\u8f93\u51fa################################################
- log4j.appender.INFO_FILE = org.apache.log4j.DailyRollingFileAppender
- #log4j.appender.INFO_FILE.Targer = /WebLogs/Info_File
- log4j.appender.INFO_FILE.File = /WebLogs/springTransation/Info_File/log.txt
- log4j.appender.INFO_FILE.Threshold = INFO
- log4j.appender.INFO_FILE.ImmediateFlush = TRUE
- log4j.appender.INFO_FILE.Append = TRUE
- log4j.appender.INFO_FILE.Encoding = UTF-8
- log4j.appender.WARN_FILE.DataPattern = '.'YYYY-MM-dd
- log4j.appender.INFO_FILE.layout = org.apache.log4j.PatternLayout
- log4j.appender.INFO_FILE.layout.ConversionPattern = [%-5p][%-22d{yyyy/MM/dd HH:mm:ss}][%l]%n%m%n
- ##################################WARN_FILE\u8f93\u51fa#################################################
- log4j.appender.WARN_FILE = org.apache.log4j.DailyRollingFileAppender
- log4j.appender.WARN_FILE.Targer = /WebLogs/Warn_File
- log4j.appender.WARN_FILE.File = /WebLogs/springTransation/Warn_File/log.txt
- log4j.appender.WARN_FILE.Threshold = WARN
- log4j.appender.WARN_FILE.ImmediateFlush = TRUE
- log4j.appender.WARN_FILE.Append = TRUE
- log4j.appender.WARN_FILE.Encoding = UTF-8
- log4j.appender.WARN_FILE.DataPattern = '.'YYYY-MM-dd
- log4j.appender.WARN_FILE.layout = org.apache.log4j.PatternLayout
- log4j.appender.WARN_FILE.layout.ConversionPattern = [%-5p][%-22d{yyyy/MM/dd HH:mm:ss}][%l]%n%m%n
- ##################################ERROR_FILE\u8f93\u51fa#################################################
- log4j.appender.ERROR_FILE = org.apache.log4j.DailyRollingFileAppender
- log4j.appender.ERROR_FILE.Targer = /WebLogs/Error_File
- log4j.appender.ERROR_FILE.File = /WebLogs/springTransation/Error_File/log.txt
- log4j.appender.ERROR_FILE.Threshold = ERROR
- log4j.appender.ERROR_FILE.ImmediateFlush = TRUE
- log4j.appender.ERROR_FILE.Append = TRUE
- log4j.appender.ERROR_FILE.Encoding = UTF-8
- log4j.appender.ERROR_FILE.DataPattern = '.'YYYY-MM-dd
- log4j.appender.ERROR_FILE.layout = org.apache.log4j.PatternLayout
- log4j.appender.ERROR_FILE.layout.ConversionPattern = [%-5p][%-22d{yyyy/MM/dd HH:mm:ss}][%l]%n%m%n
- ##################################FATAL_FILE\u8f93\u51fa################################################
- log4j.appender.FATAL_FILE = org.apache.log4j.DailyRollingFileAppender
- log4j.appender.FATAL_FILE.Targer = /WebLogs/Fatal_File
- log4j.appender.FATAL_FILE.File = /WebLogs/springTransation/Fatal_File/log.txt
- log4j.appender.FATAL_FILE.Threshold = FATAL
- log4j.appender.FATAL_FILE.ImmediateFlush = TRUE
- log4j.appender.FATAL_FILE.Append = TRUE
- log4j.appender.FATAL_FILE.Encoding = UTF-8
- log4j.appender.FATAL_FILE.DataPattern = '.'YYYY-MM-dd
- log4j.appender.FATAL_FILE.layout = org.apache.log4j.PatternLayout
- log4j.appender.FATAL_FILE.layout.ConversionPattern = [%-5p][%-22d{yyyy/MM/dd HH:mm:ss}][%l]%n%m%n
到这里我们的转账场景搭建就算完成了,下面我们就简单看一下如何进行事务管理:
Spring中的事务管理分为:1、编程是事务;2、声明式事务、3、基于注解的事务。
Spring的事务有分为:
传播行为:
PROPAGATION_REQUIRED--支持当前事务,如果当前没有事务,就新建一个事务。这是最常见的选择。
PROPAGATION_SUPPORTS--支持当前事务,如果当前没有事务,就以非事务方式执行。
PROPAGATION_MANDATORY--支持当前事务,如果当前没有事务,就抛出异常。
PROPAGATION_REQUIRES_NEW--新建事务,如果当前存在事务,把当前事务挂起。
PROPAGATION_NOT_SUPPORTED--以非事务方式执行操作,如果当前存在事务,就把当前事务挂起。
PROPAGATION_NEVER--以非事务方式执行,如果当前存在事务,则抛出异常。
隔离级别:
Serializable:最严格的级别,事务串行执行,资源消耗最大;
REPEATABLE READ:保证了一个事务不会修改已经由另一个事务读取但未提交(回滚)的数据。避免了“脏读取”和“不可重复读取”的情况,但是带来了更多的性能损失。
READ COMMITTED:大多数主流数据库的默认事务等级,保证了一个事务不会读到另一个并行事务已修改但未提交的数据,避免了“脏读取”。该级别适用于大多数系统。
Read Uncommitted:保证了读取过程中不会读取到非法数据。隔离级别在于处理多事务的并发问题。
下面我们先一起探讨一下关于编程式的事务管理,首先我们要明白事务应该加载那一层,我们知道Dao是我们的持久层,service是我们的业务层,而我们的事务应当加在业务层,而非持久层。
1、通过TransactionManager来进行我们的事务管理,修改applicationContext.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" xmlns:aop="http://www.springframework.org/schema/aop"
- xmlns:context="http://www.springframework.org/schema/context" xmlns:tx="http://www.springframework.org/schema/tx"
- xsi:schemaLocation="http://www.springframework.org/schema/beans http://www.springframework.org/schema/beans/spring-beans.xsd
- http://www.springframework.org/schema/aop http://www.springframework.org/schema/aop/spring-aop-4.3.xsd
- http://www.springframework.org/schema/context http://www.springframework.org/schema/context/spring-context-4.3.xsd
- http://www.springframework.org/schema/tx http://www.springframework.org/schema/tx/spring-tx-4.3.xsd">
- <context:property-placeholder location="classpath:db.properties"/>
- <bean id="dataSource" class="org.springframework.jdbc.datasource.DriverManagerDataSource">
- <property name="driverClassName" value="${jdbc.dirverClass}" />
- <property name="url" value="${jdbc.url}" />
- <property name="username" value="${jdbc.user}" />
- <property name="password" value="${jdbc.password}" />
- </bean>
- <bean id="blankDao" class="com.edu.hpugs.spring.BlankDaoImpl">
- <property name="dataSource" ref="dataSource"></property>
- </bean>
- <bean id="blankService" class="com.edu.hpugs.spring.BlankServiceImpl">
- <property name="blankDao" ref="blankDao"></property>
- <property name="transactionTemplate" ref="transactionTemplate"></property>
- </bean>
- <bean id="transactionManager" class="org.springframework.jdbc.datasource.DataSourceTransactionManager">
- <property name="dataSource" ref="dataSource"></property>
- </bean>
- <bean id="transactionTemplate" class="org.springframework.transaction.support.TransactionTemplate">
- <property name="transactionManager" ref="transactionManager"></property>
- </bean>
- </beans>
2、同样修改serviceImpl.java
- public class BlankServiceImpl implements IBlankService {
- private IBlankDao blankDao;
- public void setBlankDao(IBlankDao blankDao) {
- this.blankDao = blankDao;
- }
- private TransactionTemplate transactionTemplate;
- public void setTransactionTemplate(TransactionTemplate transactionTemplate) {
- this.transactionTemplate = transactionTemplate;
- }
- public void transferAccounts(final String outName, final String inName, final double monery) {
- // TODO Auto-generated method stub
- transactionTemplate.execute(new TransactionCallbackWithoutResult() {
- @Override
- protected void doInTransactionWithoutResult(TransactionStatus arg0) {
- // TODO Auto-generated method stub
- blankDao.outMonery(outName, monery);
- int i = 1 / 0;
- blankDao.inMonery(inName, monery);
- }
- });
- }
- }
到这里Spring编程式的注解就完成了。源码下载
下面我们看一下基于声明式的事务管理:
1、对于声明式的事务管理,Spring是基于AOP来实现的,所以我们需要在maven中添加aop相关的依赖
- <!-- https://mvnrepository.com/artifact/org.springframework/spring-aop -->
- <dependency>
- <groupId>org.springframework</groupId>
- <artifactId>spring-aop</artifactId>
- <version>4.3.11.RELEASE</version>
- </dependency>
- <!-- https://mvnrepository.com/artifact/org.aspectj/aspectjrt -->
- <dependency>
- <groupId>org.aspectj</groupId>
- <artifactId>aspectjrt</artifactId>
- <version>1.8.10</version>
- </dependency>
- <!-- https://mvnrepository.com/artifact/org.aspectj/aspectjweaver -->
- <dependency>
- <groupId>org.aspectj</groupId>
- <artifactId>aspectjweaver</artifactId>
- <version>1.8.10</version>
- </dependency>
2、将我们的场景代码恢复如初,然后修改我们的applicationContext.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" xmlns:aop="http://www.springframework.org/schema/aop"
- xmlns:context="http://www.springframework.org/schema/context" xmlns:tx="http://www.springframework.org/schema/tx"
- xsi:schemaLocation="http://www.springframework.org/schema/beans http://www.springframework.org/schema/beans/spring-beans.xsd
- http://www.springframework.org/schema/aop http://www.springframework.org/schema/aop/spring-aop-4.3.xsd
- http://www.springframework.org/schema/context http://www.springframework.org/schema/context/spring-context-4.3.xsd
- http://www.springframework.org/schema/tx http://www.springframework.org/schema/tx/spring-tx-4.3.xsd">
- <context:property-placeholder location="classpath:db.properties"/>
- <bean id="dataSource" class="org.springframework.jdbc.datasource.DriverManagerDataSource">
- <property name="driverClassName" value="${jdbc.dirverClass}" />
- <property name="url" value="${jdbc.url}" />
- <property name="username" value="${jdbc.user}" />
- <property name="password" value="${jdbc.password}" />
- </bean>
- <bean id="blankDao" class="com.edu.hpugs.spring.BlankDaoImpl">
- <property name="dataSource" ref="dataSource"></property>
- </bean>
- <bean id="blankService" class="com.edu.hpugs.spring.BlankServiceImpl">
- <property name="blankDao" ref="blankDao"></property>
- </bean>
- <bean id="transactionManager" class="org.springframework.jdbc.datasource.DataSourceTransactionManager">
- <property name="dataSource" ref="dataSource"></property>
- </bean>
- <!--配置事务增强(如何管理事务,只读、读写...)-->
- <tx:advice id="txAdvice" transaction-manager="transactionManager">
- <tx:attributes>
- <tx:method name="out*" propagation="REQUIRED" />
- <tx:method name="in*" propagation="REQUIRED" />
- <tx:method name="*" />
- </tx:attributes>
- </tx:advice>
- <!--aop配置,拦截哪些方法(切入点表达式,拦截上面的事务增强)-->
- <aop:config>
- <aop:pointcut id="pt" expression="execution(* com.edu.hpugs.spring.BlankServiceImpl.*(..))"/>
- <aop:advisor advice-ref="txAdvice" pointcut-ref="pt"/>
- </aop:config>
- </beans>
好了,到这里关于Spring中的声明式事务就介绍完毕。源码下载
最后时我们基于注解的事务管理
1、基于注解的事务管理,需要我们通过maven添加注解依赖:
- <!-- https://mvnrepository.com/artifact/org.springframework/spring-tx -->
- <dependency>
- <groupId>org.springframework</groupId>
- <artifactId>spring-tx</artifactId>
- <version>4.3.11.RELEASE</version>
- </dependency>
2、修改applicationContext.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" xmlns:aop="http://www.springframework.org/schema/aop"
- xmlns:context="http://www.springframework.org/schema/context" xmlns:tx="http://www.springframework.org/schema/tx"
- xsi:schemaLocation="http://www.springframework.org/schema/beans http://www.springframework.org/schema/beans/spring-beans.xsd
- http://www.springframework.org/schema/aop http://www.springframework.org/schema/aop/spring-aop-4.3.xsd
- http://www.springframework.org/schema/context http://www.springframework.org/schema/context/spring-context-4.3.xsd
- http://www.springframework.org/schema/tx http://www.springframework.org/schema/tx/spring-tx-4.3.xsd">
- <context:property-placeholder location="classpath:db.properties"/>
- <bean id="dataSource" class="org.springframework.jdbc.datasource.DriverManagerDataSource">
- <property name="driverClassName" value="${jdbc.dirverClass}" />
- <property name="url" value="${jdbc.url}" />
- <property name="username" value="${jdbc.user}" />
- <property name="password" value="${jdbc.password}" />
- </bean>
- <bean id="blankDao" class="com.edu.hpugs.spring.BlankDaoImpl">
- <property name="dataSource" ref="dataSource"></property>
- </bean>
- <bean id="blankService" class="com.edu.hpugs.spring.BlankServiceImpl">
- <property name="blankDao" ref="blankDao"></property>
- </bean>
- <bean id="transactionManager" class="org.springframework.jdbc.datasource.DataSourceTransactionManager">
- <property name="dataSource" ref="dataSource"></property>
- </bean>
- <tx:annotation-driven transaction-manager="transactionManager"/>
- </beans>
3、在需要添加事务的方法前添加注解
- public class BlankServiceImpl implements IBlankService {
- private IBlankDao blankDao;
- public void setBlankDao(IBlankDao blankDao) {
- this.blankDao = blankDao;
- }
- @Transactional
- public void transferAccounts(final String outName, final String inName, final double monery) {
- // TODO Auto-generated method stub
- blankDao.outMonery(outName, monery);
- //int i = 1 / 0;
- blankDao.inMonery(inName, monery);
- }
- }
OK,关于注解式事务的分享就到这里,源码下载。到这里关于Spring中事务的相关内容就先和大家分享到这里。感兴趣的童鞋可以留言讨论。下一篇:【Spring jdbcTemplate】
Spring事务管理(一)的更多相关文章
- 【Java EE 学习 52】【Spring学习第四天】【Spring与JDBC】【JdbcTemplate创建的三种方式】【Spring事务管理】【事务中使用dbutils则回滚失败!!!??】
一.JDBC编程特点 静态代码+动态变量=JDBC编程. 静态代码:比如所有的数据库连接池 都实现了DataSource接口,都实现了Connection接口. 动态变量:用户名.密码.连接的数据库. ...
- spring事务管理器设计思想(二)
上文见<spring事务管理器设计思想(一)> 对于第二个问题,涉及到事务的传播级别,定义如下: PROPAGATION_REQUIRED-- 如果当前没有事务,就新建一个事务.这是最常见 ...
- spring事务管理器设计思想(一)
在最近做的一个项目里面,涉及到多数据源的操作,比较特殊的是,这多个数据库的表结构完全相同,由于我们使用的ibatis框架作为持久化层,为了防止每一个数据源都配置一套规则,所以重新实现了数据源,根据线程 ...
- 事务管理(下) 配置spring事务管理的几种方式(声明式事务)
配置spring事务管理的几种方式(声明式事务) 概要: Spring对编程式事务的支持与EJB有很大的区别.不像EJB和Java事务API(Java Transaction API, JTA)耦合在 ...
- Spring事务管理器的应对
Spring抽象的DAO体系兼容多种数据访问技术,它们各有特色,各有千秋.像Hibernate是非常优秀的ORM实现方案,但对底层SQL的控制不太方便:而iBatis则通过模板化技术让你方便地控制SQ ...
- Spring事务管理(转)
1 初步理解 理解事务之前,先讲一个你日常生活中最常干的事:取钱. 比如你去ATM机取1000块钱,大体有两个步骤:首先输入密码金额,银行卡扣掉1000元钱:然后ATM出1000元钱.这两个步骤必须是 ...
- [Spring框架]Spring 事务管理基础入门总结.
前言:在之前的博客中已经说过了数据库的事务, 不过那里面更多的是说明事务的一些锁机制, 今天来说一下Spring管理事务的一些基础知识. 之前的文章: [数据库事务与锁]详解一: 彻底理解数据库事务一 ...
- Spring 事务管理 01 ——
目录: 参考: 1.Spring 事务管理高级应用难点剖析: 第 1 部分
- Spring 事务管理原理探究
此处先粘贴出Spring事务需要的配置内容: 1.Spring事务管理器的配置文件: 2.一个普通的JPA框架(此处是mybatis)的配置文件: <bean id="sqlSessi ...
- Spring 事务管理高级应用难点剖析--转
第 1 部分 http://www.ibm.com/search/csass/search/?q=%E4%BA%8B%E5%8A%A1&sn=dw&lang=zh&cc=CN& ...
随机推荐
- Vue项目搭建完整剖析全过程
Vue项目搭建完整剖析全过程 项目源码地址:https://github.com/ballyalex 有帮助的话就加个星星呗~! 项目技术栈:vue+webpack+bower+sass+axios ...
- 使用Spring Cloud和Docker构建微服务架构
原文:https://dzone.com/articles/microservice-architecture-with-spring-cloud-and-do 作者:Alexander Lukyan ...
- 自定义Git之搭建Git服务器
在远程仓库一节中,我们讲了远程仓库实际上和本地仓库没啥不同,纯粹为了7x24小时开机并交换大家的修改. GitHub就是一个免费托管开源代码的远程仓库.但是对于某些视源代码如生命的商业公司来说,既不想 ...
- 在自学java路上遇上的南墙
从2016年12月20号自学java,先是咨询了下培训中心,得小两万,四个月毕业,算了一笔账,一百二十天,合下来每天三百多块,再加上开销之类压力太大,于是开始入坑自学,随后血一般的教训直面而来: 1. ...
- Fluent Validation with Web Api 2
using FluentValidation;using FluentValidation.Attributes;using System;using System.Collections.Gener ...
- Java中容器的两种初始化方式比较
List,Set,Map的两种初始化赋值方式 List List<Integer> list2 = new ArrayList<Integer>(); for (int i= ...
- 转载的log4cplus使用指南
以下转载的log4cplus使用指南的版本可能不是最新,仅作参考了解.应以最新安装包中的示例代码为准. 目 录1 Log4cplus简介 52 安装方法 53 主要类说明 64 ...
- HTML学习笔记2
5.超链接 3种超链接: 1. 连接到其他页面 2. 锚: (是链接页面)指给超链接起一个名字,作用是连接到本页面或者其他页面的特定位置.使用name属性给超链起名字,本页要加# 3. 连接到邮件: ...
- CharMatch(括号匹配)
用自己定义的链栈实现括号匹配 #include"LinkStack.h" bool Match(char *s) { LinkStack<char> tmpS; cha ...
- JavaScript正则表达式检验与递归函数实际应用
JS递归函数(菲波那切数列) 实例解析: 一组数字:0 1 1 2 3 5 8 13 0 1 2 3 4 5 6 7 sl(0)=0; sl ...