SpringMVC和Hibernate的事物控制

2018-09-04  本文已影响0人  HitC
cute_HC

为什么使用事物

事物的存在保证了业务的原子性和隔离性。

SpringMVC配置事物的两种方式

注意点

<!-- 事物管理器配置 -->
<bean id="transactionManager" class="org.springframework.orm.hibernate4.HibernateTransactionManager">
    <property name="sessionFactory" ref="sessionFactory" />
</bean>
<!--pom添加aspect依赖-->
<dependency>
        <groupId>org.springframework</groupId>
        <artifactId>spring-aspects</artifactId>
        <version>${spring.version}</version>
</dependency>
<!--spring-aop配置-->
<tx:advice id="txAdvice" transaction-manager="txManager">
<tx:attributes>
        <!---->
        <tx:method name="find*" read-only="true" />
        <tx:method name="*" propagation ="REQUIRED" read-only="false"/>
</tx:attributes>
</tx:advice>
<aop:config>
    <aop:pointcut id="point-cut" expression="execution(* cn.orcish.service..*.*(..))"/>
    <aop:advisor advice-ref="txAdvice" pointcut-ref="point-cut"/>
</aop:config>

附:pointcut 表达式的常用写法

任意公共方法的执行:
execution(public * *(..))

任何一个名字以 set 开始的方法的执行: 
execution(* set*(..)) 

AccountService 接口定义的任意方法的执行:
execution(* com.xyz.service.AccountService.*(..)) 

在 service 包中定义的任意方法的执行:
execution(* com.xyz.service.*.*(..)) 

在 service 包或其子包中定义的任意方法的执行:
execution(* com.xyz.service..*.*(..)) 

在 service 包中的任意连接点(在 Spring AOP 中只是方法执行): 
within(com.xyz.service.*) 

在 service 包或其子包中的任意连接点(在 Spring AOP 中只是方法执行):
within(com.xyz.service..*) 

实现 AccountService 接口的代理对象的任意连接点 (在 Spring AOP 中只是方法执行):
this(com.xyz.service.AccountService) 

实现 AccountService 接口的目标对象的任意连接点 (在 Spring AOP 中只是方法执行): 
target(com.xyz.service.AccountService) 

任何一个只接受一个参数,并且运行时所传入的参数是 Serializable 接口的连接点(在 Spring AOP 中只是方法执行): 
args(java.io.Serializable) 

请注意在例子中给出的切入点不同于execution(* *(Java.io.Serializable)),args 版本只有在动态运行时候传入参数是 Serializable 时才匹配,而 execution 版本在方法签名中声明只有一个 Serializable 类型的参数时候匹配。 
目标对象中有一个 @Transactional 注解的任意连接点 (在 Spring AOP 中只是方法执行):
@target(org.springframework.transaction.annotation.Transactional) 
任何一个目标对象声明的类型有一个 @Transactional 注解的连接点 (在 Spring AOP 中只是方法执行):
@within(org.springframework.transaction.annotation.Transactional) 
任何一个执行的方法有一个 @Transactional 注解的连接点 (在 Spring AOP 中只是方法执行):
@annotation(org.springframework.transaction.annotation.Transactional) 
任何一个只接受一个参数,并且运行时所传入的参数类型具有 @Classified 注解的连接点(在 Spring AOP 中只是方法执行): 
@args(com.xyz.security.Classified) 
任何一个在名为 tradeService 的 Spring bean 之上的连接点 (在 Spring AOP 中只是方法执行): 
bean(tradeService) 
任何一个在名字匹配通配符表达式*Service的 Spring bean 之上的连接点 (在 Spring AOP 中只是方法执行):
bean(*Service) 
其中,this、tagart、args、 @target、 @with、 @annotation和@args在绑定表单中更加常用。

关于Propagation的配置详见Spring中propagation的7种事务配置
propagation=Propagation.REQUIRED:如果有事务, 那么加入事务, 没有的话新建一个(默认情况下) propagation=Propagation.NOT_SUPPORTED:容器不为这个方法开启事务 propagation=Propagation.REQUIRES_NEW:不管是否存在事务,都创建一个新的事务,原来的挂起,新的执行完毕,继续执行老的事务 propagation=Propagation.MANDATORY:必须在一个已有的事务中执行,否则抛出异常 propagation=Propagation.NEVER:必须在一个没有的事务中执行,否则抛出异常(与Propagation.MANDATORY相反) propagation=Propagation.SUPPORTS:如果其他bean调用这个方法,在其他bean中声明事务,那就用事务.如果其他bean没有声明事务,那就不用事务. propagation=Propagation.NESTED:支持当前事务,如果当前事务存在,则执行一个嵌套事务,如果当前没有事务,就新建一个事务。

添加注解扫描配置

<!--开启注解扫描-->
<tx:annotation-driven transaction-manager="transactionManager" />

注解可以使用在类和方法体上

<!--在类上使用注解-->
@Service
@Transactional(readOnly = false,propagation = Propagation.REQUIRED,rollbackFor = Exception.class)
public class BaseServiceImpl implements BaseService {
}

<!--在方法上使用注解-->
/**
* 捕获到任何异常则回滚
* Propagation.REQUIRED 如果当前没有事务,就新建一个事务,如果已经存在一个事务中,加入到这个事务中。这是最常见的选择。
*/
@Transactional(propagation= Propagation.REQUIRED,rollbackFor = Exception.class)
 public Long saveUser(User user) {
    Long l = userDao.save(user);
    String bug = null;
    //手动抛出异常
    System.out.println(bug.length());
    return l;
}
上一篇下一篇

猜你喜欢

热点阅读