spring的事物怎么配置? - 爱问答

(爱问答)

spring的事物怎么配置?

Spring事务主要有两种配置方式:一种是基于注解配置,另一种是基于配置文件。

 

1.     注解配置

1)    Spring配置文件

 

首先需要在spring配置文件当中添加事务管理容器:

 

再加入事务之前一定要先排除Controller类不被事务扫描装配,此时service还没有进行事务增强处理,得到的将是原样的Service(没有经过事务加强处理,故而没有事务处理能力),

<context:component-scanbase-package="com.syswin.crmftw.platform">

<!-- 不扫描Controller注解的类 -->

<context:exclude-filter type="annotation"expression="org.springframework.stereotype.Controller"/>

</context:component-scan>

 

<!-- 声明事务管理 -->

<bean id="txManager"  class="org.springframework.jdbc.datasource.DataSourceTransactionManager">

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

</bean>

 

<!—支持事务的注解配置 -->

<tx:annotation-driven transaction-manager="txManager"/>

 

2)    Service事务配置

Spring事务的传播行为在service类前加上@Transactional,声明这个service所有方法需要事务管理。每一个业务方法开始时都会打开一个事务。

Spring默认况下会对运行期例外(RunTimeException)进行事务回滚。这个例外是unchecked

如果遇到checked意外就不回滚。

如何改变默认规则:

1 让checked例外也回滚:在整个方法前加上 @Transactional(rollbackFor=Exception.class)

2 让unchecked例外不回滚: @Transactional(notRollbackFor=RunTimeException.class)

3 不需要事务管理的(只查询的)方法:@Transactional(propagation=Propagation.NOT_SUPPORTED)

在整个方法运行前就不会开启事务

可以加上:@Transactional(propagation=Propagation.NOT_SUPPORTED,readOnly=true),这样就做成一个只读事务,可以提高效率。

各种属性的意义:

REQUIRED:业务方法需要在一个容器里运行。如果方法运行时,已经处在一个事务中,那么加入到这个事务,否则自己新建一个新的事务。

NOT_SUPPORTED:声明方法不需要事务。如果方法没有关联到一个事务,容器不会为他开启事务,如果方法在一个事务中被调用,该事务会被挂起,调用结束后,原先的事务会恢复执行。

REQUIRESNEW:不管是否存在事务,该方法总汇为自己发起一个新的事务。如果方法已经运行在一个事务中,则原有事务挂起,新的事务被创建。

MANDATORY:该方法只能在一个已经存在的事务中执行,业务方法不能发起自己的事务。如果在没有事务的环境下被调用,容器抛出例外。

SUPPORTS:该方法在某个事务范围内被调用,则方法成为该事务的一部分。如果方法在该事务范围外被调用,该方法就在没有事务的环境下执行。

NEVER:该方法绝对不能在事务范围内执行。如果在就抛例外。只有该方法没有关联到任何事务,才正常执行。

NESTED:如果一个活动的事务存在,则运行在一个嵌套的事务中。如果没有活动事务,则按REQUIRED属性执行。它使用了一个单独的事务,这个事务拥有多个可以回滚的保存点。内部事务的回滚不会对外部事务造成影响。它只对DataSourceTransactionManager事务管理器起效。

例如:

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

public void saveCenter(Center center) {

         centerDao.saveCenter(center);

         throw newRecoverableDataAccessException("测试异常回滚操作");

}

此时保存操作将会进行回滚操作。也可以直接@Transactional选择它的默认操作。

Transactional注解类源码如下:

@Target({ElementType.METHOD, ElementType.TYPE})

@Retention(RetentionPolicy.RUNTIME)

@Inherited

@Documented

public @interface Transactional {

         String value() default"";

Propagationpropagation() default Propagation.REQUIRED;

Isolationisolation() default Isolation.DEFAULT;

int timeout()default TransactionDefinition.TIMEOUT_DEFAULT;

booleanreadOnly() default false;

Class<?extends Throwable>[] rollbackFor() default {};

String[]rollbackForClassName() default {};

Class<? extendsThrowable>[] noRollbackFor() default {};

String[]noRollbackForClassName() default {};

}

1.     配置文件

<!-- 配置事务管理器-->

<bean id="transactionManager"

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

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

</bean>

 

<!--声明式容器事务管理,transaction-manager指定事务管理器为transactionManager -->

    <tx:advice id="txAdvice"transaction-manager="transactionManager">

        <tx:attributes>

            <tx:method name="add*"propagation="REQUIRED" />

            <tx:method name="get*"propagation="REQUIRED" />

            <tx:method name="*"read-only="true" />

        </tx:attributes>

    </tx:advice>

   

    <aop:configexpose-proxy="true">

        <!-- 只对业务逻辑层实施事务 -->

        <aop:pointcut id="txPointcut"

expression="execution(*com.syswin.crmftw.platform.center.service.*.*(..))"/>

        <!-- Advisor定义,切入点和通知分别为txPointcut、txAdvice-->

        <aop:advisorpointcut-ref="txPointcut" advice-ref="txAdvice"/>

</aop:config>

 

其中主要配置中是tx:advice和aop:config两个配置节,以SpringAOP的方式实现事务管理。

tx:advice配置了事务的管理者是transactionManager,同时tx:method也规定了如果方法名匹配“add*”和“get*”方法时使用事务,propagation是设定事务的传播级别。除了“add*”和“get*”方法,其他的方法的事务是只读的(典型地,对于只执行查询的事务你会将该属性设为true,如果出现了更新、插入或是删除语句时只读事务就会失败)

aop:config指定了一个aop:pointcut去引用上边的advice。

这样就通过AOP的拦截机制实现了事务.

2.     Spring AOP expression表达式

例如:

         <aop:configexpose-proxy="true">

       <aop:pointcutid="txPointcut"

expression="execution(*com.syswin.crmftw.platform.center.service.*.*(..))" />

               <aop:advisorpointcut-ref="txPointcut" advice-ref="txAdvice"/>

</aop:config>          

在上面代码中:

execution   是方法运行

第一个“*” 是指定public的方法

com.syswin.crmftw.platform.center.service.*.*  是指定目录下的所有类所有的方法

com.syswin.crmftw.platform.center.service.insert*是指定目录下的指定类insert开头的任意方法

(..)      是指任何参数,可以没有参数

        

在execution中是可以有多个的方法,例如:

execution(*com.action.userinfoAction..*(..))

&&execution(*com.action.memberAction..*(..))

&&!execution(*get*(..))&&!execution(* set*(..))


不好搞。春天的什么事物了?太多了...

相关标签:配置

下一篇:for循环中return,countinue,break的作用

上一篇:这个符号怎么弄出来?

热门标签:
excel 网盘 破解 word dll
最新更新:
微软重新评估新的Outlook的使用时机 联想推出搭载联发科Helio G80芯片组的Tab M9平板 英特尔创新大赛时间确定! 微软Edge浏览器在稳定渠道中推出Workspaces功能 英伟达RTX4060TiGPU推出MaxSun动漫主题! 谷歌地图为用户提供了街景服务! GameSir 在T4 Kaleid中推出了一款出色的控制器! 微软开始在Windows 11 中测试其画图应用程序的新深色模式! LG电子推出全球首款无线OLED电视 英伟达人工智能芯片崭露头角! Steam Deck可以玩什么游戏-Steam Deck价格限时优惠 雷蛇推出CobraPro鼠标 Kindle电子阅读器可以访问谷歌商店吗 Windows10如何加入组策略 window10图片查看器怎么没有了?