Existem duas maneiras de gerenciar transações da Spring: uma é a tradicional gestão programática de transações, ou seja, a administração de transações realizada por código, a outra é a gestão declarativa de transações baseada na tecnologia AOP. Como o gerenciamento programático de transações raramente é usado no desenvolvimento real, explicaremos apenas o gerenciamento declarativo de transações do Spring em detalhes.
A seguir, é apresentada uma explicação detalhada da implementação da transação a partir do método de acesso a dados:
<!-- 加载配置文件 数据源db.properties -->
<context:property-placeholder location="classpath:config/db.properties"/>
<!-- 数据源配置,使用数据库连接池druid -->
<bean id="dataSource" class="com.alibaba.druid.pool.DruidDataSource"
init-method="init" destroy-method="close">
<!-- 数据库驱动 -->
<property name="driverClassName" value="${jdbc.driver}" />
<!-- 基本属性 url、user、password -->
<property name="url" value="${jdbc.url}" />
<property name="username" value="${jdbc.username}" />
<property name="password" value="${jdbc.password}" />
<!-- 配置初始化大小、最小、最大 -->
<property name="initialSize" value="1" />
<property name="minIdle" value="1" />
<property name="maxActive" value="20" />
<!-- 配置获取连接等待超时的时间 -->
<property name="maxWait" value="60000" />
<!-- 配置间隔多久才进行一次检测,检测需要关闭的空闲连接,单位是毫秒 -->
<property name="timeBetweenEvictionRunsMillis" value="60000" />
<!-- 配置一个连接在池中最小生存的时间,单位是毫秒 -->
<property name="minEvictableIdleTimeMillis" value="300000" />
<!-- 打开PSCache,并且指定每个连接上PSCache的大小 -->
<property name="poolPreparedStatements" value="true" />
<property name="maxPoolPreparedStatementPerConnectionSize"
value="20" />
<!-- 配置监控统计拦截的filters,去掉后监控界面sql无法统计 记录日志-->
<property name="filters" value="stat,slf4j" />
</bean>
Implementação de gerenciamento de transações JDBC:
<bean id="jdbcTemplate" class="org.springframework.jdbc.core.JdbcTemplate">
<property name="dataSource" ref="dataSource"/>
</bean>
<bean id="transactionManager" class="org.springframework.jdbc.datasource.DataSourceTransactionManager">
<property name="dataSource" ref="dataSource"/>
</bean>
Implementação de gerenciamento de transações Mybatis:
<!-- spring集成mybatis集中管理session -->
<bean id="sqlSessionFactory" class="org.mybatis.spring.SqlSessionFactoryBean">
<!-- 引入数据源 -->
<property name="dataSource" ref="dataSource"/>
<!-- 加载mybatis核心配置文件-->
<property name="configlocation" value="classpath:config/mybatis.cfg.xml"/>
<!-- 批量别名定义,自动扫描包,自动定义别名,别名就是类名(首字母大写或小写都可以) -->
<!-- <property name="typeAliasesPackage" value="com.huayuan.bean"></property> -->
<!-- 扫描sql配置文件,mapper.xml -->
<!-- <property name="mapperLocations" value="classpath:com/huayuan/mapper/*.xml"></property> -->
</bean>
<!-- 配置Ibatis的事务管理器 -->
<bean id="transactionManager" class="org.springframework.jdbc.datasource.DataSourceTransactionManager">
<property name="dataSource" ref="dataSource"/>
</bean>
Implementação de gerenciamento de transação de hibernação:
<!-- 配置hibernate的sessionFactory -->
<bean id="sessionFactory" class="org.springframework.orm.hibernate3.LocalSessionFactoryBean">
<property name="dataSource">
<ref bean="dataSource" />
</property>
<property name="hibernateProperties">
<props>
<prop key="hibernate.dialect">
org.hibernate.dialect.MySQLDialect
</prop>
<prop key="hibernate.show_sql">true</prop>
</props>
</property>
<property name="mappingResources">
<list>
<value>com/hisu/beans/User.hbm.xml</value>
</list>
</property>
</bean>
A seguir, é apresentado um método geral para implementar transações:
1. Cada Bean possui um proxy:
<!-- 配置DAO -->
<bean id="userDAO" class="com.hisu.DAOImpl.UserDAOImpl" scope="singleton">
<property name="sessionFactory" ref="sessionFactory" />
</bean>
<bean id="userDAOProxy"
class="org.springframework.transaction.interceptor.TransactionProxyFactoryBean" lazy-init="true" abstract="true">
<property name="transactionManager" ref="transactionManager" />
<property name="target" ref="userDAO" />
<!-- 配置事务属性 -->
<property name="transactionAttributes">
<props>
<prop key="insert*">PROPAGATION_REQUIRED</prop>
<prop key="update*">PROPAGATION_REQUIRED</prop>
<prop key="delete*">PROPAGATION_REQUIRED</prop>
<prop key="get*">PROPAGATION_REQUIRED,readOnly</prop>
</props>
</property>
</bean>
Segundo, todos os beans compartilham uma classe base de proxy:
<bean id="transactionBase"
class="org.springframework.transaction.interceptor.TransactionProxyFactoryBean" lazy-init="true" abstract="true">
<property name="transactionManager" ref="transactionManager" />
<!-- 配置事务属性 -->
<property name="transactionAttributes">
<props>
<prop key="insert*">PROPAGATION_REQUIRED</prop>
<prop key="update*">PROPAGATION_REQUIRED</prop>
<prop key="delete*">PROPAGATION_REQUIRED</prop>
<prop key="get*">PROPAGATION_REQUIRED,readOnly</prop>
</props>
</property>
</bean>
<!-- 配置DAO -->
<bean id="userDAO" class="com.hisu.DAOImpl.UserDAOImpl" parent="transactionBase" scope="singleton">
<property name="sessionFactory" ref="sessionFactory" />
</bean>
Terceiro, use o interceptador:
<bean id="transactionInterceptor"
class="org.springframework.transaction.interceptor.TransactionInterceptor" lazy-init="true" abstract="true">
<property name="transactionManager" ref="transactionManager" />
<!-- 配置事务属性 -->
<property name="transactionAttributes">
<props>
<prop key="insert*">PROPAGATION_REQUIRED</prop>
<prop key="update*">PROPAGATION_REQUIRED</prop>
<prop key="delete*">PROPAGATION_REQUIRED</prop>
<prop key="get*">PROPAGATION_REQUIRED,readOnly</prop>
</props>
</property>
</bean>
<bean class="org.springframework.aop.framework.autoproxy.BeanNameAutoProxyCreator">
<property name="beanNames">
<list>
<value>com.hisu.dao.*</value>
</list>
</property>
<property name="interceptorNames">
<list>
<value>transactionInterceptor</value>
</list>
</property>
</bean>
4. Interceptores configurados com tags tx: (recomendado)
<!-- 配置事务的传播特性 -->
<tx:advice id="txAdvice" transaction-manager="transactionManager">
<tx:attributes>
<tx:method name="add*" propagation="REQUIRED"/>
<tx:method name="delete*" propagation="REQUIRED"/>
<tx:method name="modify*" propagation="REQUIRED"/>
<tx:method name="*" read-only="true"/>
</tx:attributes>
</tx:advice>
<!-- 那些类的哪些方法参与事务 -->
<aop:config>
<aop:advisor pointcut="execution(* com.evan.crm.service.*.*(..))" advice-ref="txAdvice"/>
</aop:config>
(* com.evan.crm.service. *. * (..)) O significado de vários curingas:
O primeiro tipo de valor de retorno
curinga * -qualquer classe
* -qualquer pacote curinga com.evan.crm.service terceiro * - o pacote curinga com.evan.crm.service Qualquer método de qualquer classe .. O
quarto ..- O método curinga pode ter 0 ou mais parâmetros
Cinco anotações completas (manipulam automaticamente interfaces, classes, métodos marcados como "@Transaction")
<tx:annotation-driven transaction-manager="transactionManager"/>