spring+mybatis之注解式事务管理初识(小实例)

本文内容纲要:spring+mybatis之注解式事务管理初识(小实例)

1.上一章,我们谈到了spring+mybatis声明式事务管理,我们在文章末尾提到,在实际项目中,用得更多的是注解式事务管理,这一章将学习一下注解式事务管理的有关知识.注解式事务管理只需要在上一节的小实例上更改两个文件就好:

  (1)更改spring配置文件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: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-3.0.xsd

http://www.springframework.org/schema/context

http://www.springframework.org/schema/context/spring-context-3.0.xsd

http://www.springframework.org/schema/tx

http://www.springframework.org/schema/tx/spring-tx-3.0.xsd ">

<!-- 加载配置文件 -->

<context:property-placeholder location="config.properties" />

<!-- 指定spring注解注入层 -->

<context:component-scan base-package="com.gnc" />

<!-- 数据库连接池管理 -->

<bean id="c3p0DataSource" class="com.mchange.v2.c3p0.ComboPooledDataSource"

destroy-method="close">

<property name="driverClass" value="${db.driverClass}"></property>

<property name="jdbcUrl" value="${db.jdbcUrl}"></property>

<property name="user" value="${db.user}"></property>

<property name="password" value="${db.password}"></property>

<property name="initialPoolSize" value="${db.initialPoolSize}"></property>

<!--最大空闲时间,60秒内未使用则连接被丢弃。若为0则永不丢弃。Default: 0 -->

<property name="maxIdleTime" value="${db.maxIdleTime}"></property>

<!--连接池中保留的最大连接数。Default: 15 -->

<property name="maxPoolSize" value="${db.maxPoolSize}"></property>

<property name="minPoolSize" value="${db.minPoolSize}"></property>

<!--当连接池中的连接耗尽的时候c3p0一次同时获取的连接数。Default: 3 -->

<property name="acquireIncrement" value="${db.acquireIncrement}"></property>

<!--两次连接中间隔时间,单位毫秒。Default: 1000 -->

<property name="acquireRetryDelay" value="${db.acquireRetryDelay}"></property>

<!--定义在从数据库获取新连接失败后重复尝试的次数。Default: 30 -->

<property name="acquireRetryAttempts" value="${db.acquireRetryAttempts}"></property>

<!--获取连接失败将会引起所有等待连接池来获取连接的线程抛出异常。但是数据源仍有效保留,并在下次调用getConnection()的时候继续尝试获取连接。如果设为true,那么在尝试

获取连接失败后该数据源将申明已断开并永久关闭。Default: false -->

<property name="breakAfterAcquireFailure" value="${db.breakAfterAcquireFailure}"></property>

</bean>

<!-- ================================事务相关控制================================================= -->

<bean name="transactionManager"

class="org.springframework.jdbc.datasource.DataSourceTransactionManager">

<property name="dataSource" ref="c3p0DataSource"></property>

</bean>

<tx:annotation-driven transaction-manager="transactionManager"

proxy-target-class="true" />

<bean id="sqlSessionFactory" class="org.mybatis.spring.SqlSessionFactoryBean">

<property name="dataSource" ref="c3p0DataSource" />

<property name="configLocation" value="MyBatis-Configuration.xml" />

</bean>

<bean class="org.mybatis.spring.mapper.MapperFactoryBean">

<property name="mapperInterface" value="com.gnc.mapper.UserMapper" />

<property name="sqlSessionFactory" ref="sqlSessionFactory" />

</bean>

</beans>

  (2)更改UserServiceImpl.java实现类:

  

package com.gnc.serviceImpl;

import java.util.Map;

import org.springframework.beans.factory.annotation.Autowired;

import org.springframework.stereotype.Service;

import org.springframework.transaction.annotation.Propagation;

import org.springframework.transaction.annotation.Transactional;

import com.gnc.dao.UserDao;

import com.gnc.model.User;

import com.gnc.service.UserService;

@Service("userService")

public class UserServiceImpl implements UserService {

@Autowired

private UserDao userDao;

@Transactional(propagation = Propagation.SUPPORTS)

public int countAll() {

return this.userDao.countAll();

}

@Transactional(propagation = Propagation.REQUIRED , readOnly = false)

@Override

public void insertUser(User user) {

this.userDao.insertUser(user);

throw new RuntimeException("Error");

}

@Transactional(propagation = Propagation.REQUIRED , readOnly = false)

@Override

public void update_insert(Map map, User user) {

this.userDao.updateUser(map);

this.userDao.insertUser(user);

throw new RuntimeException("Error");

}

}

2.注解式事务管理相对声明式事务管理显得更加灵活,我们只需要在需要进行事务管理的方法或者类前加上@Transactional+相关控制参数即可,部分常用参数如下表:

参 数 名 称

功 能 描 述

rollbackForClassName

该属性用于设置需要进行回滚的异常类名称数组,当方法中抛出指定异常名称数组中的异常时,则进行事务回滚。例如:

指定单一异常类名称:@Transactional(rollbackForClassName="RuntimeException")

指定多个异常类名称:@Transactional(rollbackForClassName={"RuntimeException","Exception"})

noRollbackFor

该属性用于设置不需要进行回滚的异常类数组,当方法中抛出指定异常数组中的异常时,不进行事务回滚。例如:

指定单一异常类:@Transactional(noRollbackFor=RuntimeException.class)

指定多个异常类:@Transactional(noRollbackFor={RuntimeException.class, Exception.class})

noRollbackForClassName

该属性用于设置不需要进行回滚的异常类名称数组,当方法中抛出指定异常名称数组中的异常时,不进行事务回滚。例如:

指定单一异常类名称:@Transactional(noRollbackForClassName="RuntimeException")

指定多个异常类名称:

@Transactional(noRollbackForClassName={"RuntimeException","Exception"})

propagation

该属性用于设置事务的传播行为,具体取值可参考表6-7。

例如:@Transactional(propagation=Propagation.NOT_SUPPORTED,readOnly=true)

isolation

该属性用于设置底层数据库的事务隔离级别,事务隔离级别用于处理多事务并发的情况,通常使用数据库的默认隔离级别即可,基本不需要进行设置

timeout

该属性用于设置事务的超时秒数,默认值为-1表示永不超时

readOnly

该属性用于设置当前事务是否为只读事务,设置为true表示只读,false则表示可读写,默认值为false。例如:@Transactional(readOnly=true)

rollbackFor

该属性用于设置需要进行回滚的异常类数组,当方法中抛出指定异常数组中的异常时,则进行事务回滚。例如:

指定单一异常类:@Transactional(rollbackFor=RuntimeException.class)

指定多个异常类:@Transactional(rollbackFor={RuntimeException.class, Exception.class})

需要注意的几点:

1 @Transactional 只能被应用到public方法上, 对于其它非public的方法,如果标记了@Transactional也不会报错,但方法没有事务功能.

2用 spring 事务管理器,由spring来负责数据库的打开,提交,回滚.默认遇到运行期例外(throw new RuntimeException("注释");)会回滚,即遇到不受检查(unchecked)的例外时回滚;而遇到需要捕获的例外(throw new Exception("注释");)不会回滚,即遇到受检查的例外(就是非运行时抛出的异常,编译器会检查到的异常叫受检查例外或说受检查异常)时,需我们指定方式来让事务回滚 要想所有异常都回滚,要加上 @Transactional( rollbackFor={Exception.class,其它异常}) .如果让unchecked例外不回滚: @Transactional(notRollbackFor=RunTimeException.class)

如下:

@Transactional(rollbackFor=Exception.class) //指定回滚,遇到异常Exception时回滚

public void methodName() {

throw new Exception("注释");

}

@Transactional(noRollbackFor=Exception.class)//指定不回滚,遇到运行期例外(throw new RuntimeException("注释");)会回滚

public ItimDaoImpl getItemDaoImpl() {

throw new RuntimeException("注释");

}

3、@Transactional 注解可以被应用于接口定义和接口方法、类定义和类的 public 方法上。然而,请注意仅仅 @Transactional 注解的出现不足于开启事务行为,它仅仅 是一种元数据,能够被可以识别 @Transactional 注解和上述的配置适当的具有事务行为的beans所使用。上面的例子中,其实正是 tx:annotation-driven/元素的出现 开启 了事务行为。

4、Spring团队的建议是你在具体的类(或类的方法)上使用 @Transactional 注解,而不要使用在类所要实现的任何接口上。你当然可以在接口上使用 @Transactional 注解,但是这将只能当你设置了基于接口的代理时它才生效。因为注解是 不能继承 的,这就意味着如果你正在使用基于类的代理时,那么事务的设置将不能被基于类的代理所识别,而且对象也将不会被事务代理所包装(将被确认为严重的)。因 此,请接受Spring团队的建议并且在具体的类上使用 @Transactional 注解。

本文内容总结:spring+mybatis之注解式事务管理初识(小实例)

原文链接:https://www.cnblogs.com/geningchao/p/6632283.html

以上是 spring+mybatis之注解式事务管理初识(小实例) 的全部内容, 来源链接: utcz.com/z/362757.html

回到顶部