MyBatis及Spring事务初学总结
2015-07-07 12:45
537 查看
MyBatis是从iBatis项目继承而来,是目前互联网公司的主流数据访问层框架。
在实际使用中与Spring一起使用。MyBatis出了MyBatis-Spring组件,用于两者之间的整合,使用Spring的事务管理功能。
DataSourceTransactionManager是Spring框架的组件,通过java.sql.connection来管理事务(通过commit()提交事务,或者通过rollback()回滚事务)
使用MyBatis的事务有两种方式:1.注解驱动;2.XML驱动
dataSource是通过dataSource.properties来进行配置的(这里有疑问,这样实现的话,spring的事务框架连接数据库时使用的就是com.mysql.jdbc.Driver,就没有连接池管理功能了吧?)
一般如下配置
而mybatis-config.xml中一般配置的是
而userMapper.xml一般如下,同时需要一个interface,定义这些方法。
这其中的sqlSessionFactory是用Spring的SqlSessionFactoryBean来配置创建的,但是一般不直接使用,而是将其注入到了org.mybatis.spring.mapper.MapperFactoryBean中,或者其他扩展了SqlSessionDaoSupport的DAO中。
SqlSessionTemplate是MyBatis-Spring的核心类,负责管理MyBatis的SqlSession,调用MyBatis的SQL方法,翻译异常等,但是
为了代替手动使用SqlSessionDaoSupport或SqlSessionTemplate,MyBatis-Spring提供了一个动态代理的实现,即MapperFactoryBean,从而直接注入数据映射器到Service层bean中,MapperFactoryBean负责处理SqlSession的创建和关闭,使用了Spring事务后,事务完成时,session将会提交或者回滚。
其中事务的传播行为有7种,比较常见的是REQUIRED、SUPPORTS、REQUIRES_NEW、NESTED。可参考这里的图示http://docs.spring.io/spring/docs/current/spring-framework-reference/html/transaction.html#tx-propagation
具体列出如下
PROPAGATION_MANDATORY 强制的,表示该方法必须运行在一个事务中,否则抛出异常
PROPAGATION_NESTED 嵌套的,若当前已经有一个事务,则这个方法在其中在运行于一个嵌套其中的事务,可独立的提交或回滚
PROPAGATION_NEVER 不应该运行在事务中,如果一个事务正在运行则抛出异常
PROPAGATION_NOT_SUPPORTED 不应该运行在事务中,如果一个事务正在运行,则将该事务挂起
PROPAGATION_SUPPORTS 不需要事务上下文,但是也支持,如果有一个事务已经在运行,则该方法也可以在事务中运行
PROPAGATION_REQUIRES_NEW 需要新的事务。表示需要一个新的事务,如果一个现有事务正在运行,则该现有事务将被挂起
PROPAGATION_REQUIRED 需要事务,如果现在一个现有事务正在运行,则该方法直接在该事务中运行,否则起个新的事务
事务的隔离级别5种,用于说明受其他并发的事务的影响程度,通过锁表完全隔离当然可以做到没有影响,但这种情况下性能损失太大,并不适用所有场景。
ISOLATION_DEFAULT 默认的,适用数据库后端设置
ISOLATION_READ_UNCOMMITTED 允许读取尚未提交的更改。可能导致Dirty read、Phantom reads或Nonrepeatable read。
ISOLATION_READ_COMMITTED 允许从已经提交的并发事务读取,不会出现Dirty read,但是会出现Phantom reads或Nonrepeatable read。
ISOLATION_REPEATABLE_READ 对相同字段的多次读取需一直。不会出现Dirty read或Nonrepeatable read,但是还是可能出现Phantom read。
ISOLATION_SERIALIZABLE 完全服从ACID(原子性、一致性、隔离性、持久性)原则,确保不会出现Dirty read、Nonrepeatable read或Phantom read。
事务超时
事务只读
事务回滚规则:默认情况下只对继承自RuntimeException的异常(Unckecked Exception)进行回滚。
这里可以参考这几个链接,
https://github.com/makersoft/mybatis-shards
其中CobarClient是阿里的解决方案
ShardBatis:http://code.google.com/p/shardbatis/
参考资料:
1.《Spring in action》
2.https://mybatis.github.io/mybatis-3/index.html
3.https://guptavikas.wordpress.com/2010/04/15/aspectj-pointcut-expressions/
4.https://www.ibm.com/developerworks/cn/education/opensource/os-cn-spring-trans/
5.http://m.oschina.net/blog/120707
6.http://gemantic.iteye.com/blog/1622799
7.http://www.douban.com/note/227015659/
在实际使用中与Spring一起使用。MyBatis出了MyBatis-Spring组件,用于两者之间的整合,使用Spring的事务管理功能。
92 <dependency> 93 <groupId>org.mybatis</groupId> 94 <artifactId>mybatis-spring</artifactId> 95 <version>1.2.2</version> 96 </dependency>
DataSourceTransactionManager是Spring框架的组件,通过java.sql.connection来管理事务(通过commit()提交事务,或者通过rollback()回滚事务)
使用MyBatis的事务有两种方式:1.注解驱动;2.XML驱动
38 <!-- (事务管理)transaction manager, use JtaTransactionManager for global tx --> 39 <bean id="transactionManager" 40 class="org.springframework.jdbc.datasource.DataSourceTransactionManager"> 41 <property name="dataSource" ref="dataSource" /> 42 </bean> 43 44 <tx:advice id="userTxAdvice" transaction-manager="transactionManager"> 45 <tx:attributes> 46 <tx:method name="delete*" propagation="REQUIRED" read-only="false" 47 rollback-for="java.lang.Exception" no-rollback-for="java.lang.RuntimeException" /> 48 <tx:method name="insert*" propagation="REQUIRED" read-only="false" 49 rollback-for="java.lang.Exception" /> 50 <tx:method name="update*" propagation="REQUIRED" read-only="false" 51 no-rollback-for="java.lang.RuntimeException" rollback-for="java.lang.Exception"/> 52 <tx:method name="find*" propagation="SUPPORTS" /> 53 <tx:method name="get*" propagation="SUPPORTS" /> 54 <tx:method name="select*" propagation="SUPPORTS" /> 55 <tx:method name="*Transaction*" no-rollback-for="java.lang.RuntimeException" rollback-for="java.lang.Exception"/> 56 </tx:attributes> 57 </tx:advice> 58 59 <aop:config> 60 <aop:pointcut id="pc" 61 expression="execution(public * com.service.impl.*.*(..))" /> <!--把事务控制在Service层 --> 62 <aop:advisor pointcut-ref="pc" advice-ref="userTxAdvice" /> 63 </aop:config> 64 65 66 <!-- 创建SqlSessionFactory,同时指定数据源 --> 67 <bean id="sqlSessionFactory" class="org.mybatis.spring.SqlSessionFactoryBean"> 68 <property name="configLocation" value="/WEB-INF/mybatis-config.xml"></property> 69 <property name="dataSource" ref="dataSource" /> 70 </bean>
dataSource是通过dataSource.properties来进行配置的(这里有疑问,这样实现的话,spring的事务框架连接数据库时使用的就是com.mysql.jdbc.Driver,就没有连接池管理功能了吧?)
一般如下配置
driver=com.mysql.jdbc.Driver url=jdbc:mysql://127.0.0.1:3306/myfirstbase username=XXXX password=XXXX
而mybatis-config.xml中一般配置的是
1 <?xml version="1.0" encoding="UTF-8"?> 2 <!DOCTYPE configuration 3 PUBLIC "-//mybatis.org//DTDConfig 3.0//EN" 4 "http://mybatis.org/dtd/mybatis-3-config.dtd"> 5 6 <!-- mybatis的数据类型和处理器等等...都可以实现接口重写,然后配置 --> 7 <configuration> 8 9 <typeAliases> 10 <package name="com.model" /> 11 </typeAliases> 12 13 <mappers> 14 <!--映射文件--> 15 <mapper resource="com/mapper/userMapper.xml"/> 16 </mappers> 17 </configuration>
而userMapper.xml一般如下,同时需要一个interface,定义这些方法。
<?xml version="1.0" encoding="UTF-8"?> 2 <!DOCTYPE mapper 3 PUBLIC "-//mybatis.org//DTD Mapper 3.0//EN" 4 "http://mybatis.org/dtd/mybatis-3-mapper.dtd"> 5 6 7 <mapper namespace="com.mapper.IUserMapper"> 8 9 <select id="selectAllUser" resultMap="UserMapper"> 10 select * from TEST_LEE 11 </select> 12 13 <resultMap type="User" id="UserMapper" autoMapping="true"> 14 <id property="id" column="id"/> 15 <result property="name" column="name"/> 16 <result property="user_id" column="user_id"/> 17 </resultMap> 18 19 <select id="selectById" parameterType="int" resultType="User"> 20 select * from TEST_LEE where id=#{id} 21 </select> 22 23 24 <insert id="insertUser" parameterType="User"> 25 insert into TEST_LEE (user_id, name) values(#{user_id},#{name}) 26 </insert> 27 28 <update id="updateUser" parameterType="User"> 29 update TEST_LEE set name=#{name}, user_id=#{user_id} where id=#{id} 30 </update> 31 32 <delete id="deleteUser" parameterType="int"> 33 delete from TEST_LEE where id=#{id} 34 </delete> 35 36 <!-- 返回自增ID会赋值到User.id --> 37 <insert id="insertResultGeneratedKey" parameterType="User" useGeneratedKeys="true" keyProperty="id"> 38 insert into TEST_LEE (user_id, name) values(#{user_id},#{name}) 39 </insert> 40 41 </mapper>
这其中的sqlSessionFactory是用Spring的SqlSessionFactoryBean来配置创建的,但是一般不直接使用,而是将其注入到了org.mybatis.spring.mapper.MapperFactoryBean中,或者其他扩展了SqlSessionDaoSupport的DAO中。
109 <!-- Mapper接口所在包名,Spring会自动查找其下的Mapper --> 110 <bean class="org.mybatis.spring.mapper.MapperScannerConfigurer"> 111 <property name="basePackage" value="com.mapper" /> 112 </bean> 113 114 <bean id="userMapper" class="org.mybatis.spring.mapper.MapperFactoryBean"> 115 <property name="mapperInterface" value="com.mapper.IUserMapper" /> 116 <property name="sqlSessionFactory" ref="sqlSessionFactory"></property> 117 </bean>
SqlSessionTemplate是MyBatis-Spring的核心类,负责管理MyBatis的SqlSession,调用MyBatis的SQL方法,翻译异常等,但是
为了代替手动使用SqlSessionDaoSupport或SqlSessionTemplate,MyBatis-Spring提供了一个动态代理的实现,即MapperFactoryBean,从而直接注入数据映射器到Service层bean中,MapperFactoryBean负责处理SqlSession的创建和关闭,使用了Spring事务后,事务完成时,session将会提交或者回滚。
2.Spring的事务的传播行为和隔离级别
提及Spring中的声明式事务,通常提及特性包括:事务的传播行为、隔离级别、只读、事务超时、回滚规则,可以参见上面的规则定义。其中事务的传播行为有7种,比较常见的是REQUIRED、SUPPORTS、REQUIRES_NEW、NESTED。可参考这里的图示http://docs.spring.io/spring/docs/current/spring-framework-reference/html/transaction.html#tx-propagation
具体列出如下
PROPAGATION_MANDATORY 强制的,表示该方法必须运行在一个事务中,否则抛出异常
PROPAGATION_NESTED 嵌套的,若当前已经有一个事务,则这个方法在其中在运行于一个嵌套其中的事务,可独立的提交或回滚
PROPAGATION_NEVER 不应该运行在事务中,如果一个事务正在运行则抛出异常
PROPAGATION_NOT_SUPPORTED 不应该运行在事务中,如果一个事务正在运行,则将该事务挂起
PROPAGATION_SUPPORTS 不需要事务上下文,但是也支持,如果有一个事务已经在运行,则该方法也可以在事务中运行
PROPAGATION_REQUIRES_NEW 需要新的事务。表示需要一个新的事务,如果一个现有事务正在运行,则该现有事务将被挂起
PROPAGATION_REQUIRED 需要事务,如果现在一个现有事务正在运行,则该方法直接在该事务中运行,否则起个新的事务
PROPAGATION_NESTED can only be used with a DataSourceTransactionManager and a JDBC3 driver. It uses save points in order to be able to rollback some part of a transaction (i.e. what constitutes the nested transaction in Spring terms). See the javadoc of Connection to see how save points work.
事务的隔离级别5种,用于说明受其他并发的事务的影响程度,通过锁表完全隔离当然可以做到没有影响,但这种情况下性能损失太大,并不适用所有场景。
ISOLATION_DEFAULT 默认的,适用数据库后端设置
ISOLATION_READ_UNCOMMITTED 允许读取尚未提交的更改。可能导致Dirty read、Phantom reads或Nonrepeatable read。
ISOLATION_READ_COMMITTED 允许从已经提交的并发事务读取,不会出现Dirty read,但是会出现Phantom reads或Nonrepeatable read。
ISOLATION_REPEATABLE_READ 对相同字段的多次读取需一直。不会出现Dirty read或Nonrepeatable read,但是还是可能出现Phantom read。
ISOLATION_SERIALIZABLE 完全服从ACID(原子性、一致性、隔离性、持久性)原则,确保不会出现Dirty read、Nonrepeatable read或Phantom read。
事务超时
事务只读
事务回滚规则:默认情况下只对继承自RuntimeException的异常(Unckecked Exception)进行回滚。
3.MyBatis涉及到分库分表怎么办
分库分表要解决的问题是因数据库存储量大了和访问量大了进行的水平扩展问题这里可以参考这几个链接,
https://github.com/makersoft/mybatis-shards
其中CobarClient是阿里的解决方案
ShardBatis:http://code.google.com/p/shardbatis/
参考资料:
1.《Spring in action》
2.https://mybatis.github.io/mybatis-3/index.html
3.https://guptavikas.wordpress.com/2010/04/15/aspectj-pointcut-expressions/
4.https://www.ibm.com/developerworks/cn/education/opensource/os-cn-spring-trans/
5.http://m.oschina.net/blog/120707
6.http://gemantic.iteye.com/blog/1622799
7.http://www.douban.com/note/227015659/
相关文章推荐
- javaproject积累——java 反射 invoke
- Java基础日记———Collection
- java类和对象
- Spring Data JDBC Extensions - Core JDBC Extensions (概述)
- 配置Java环境变量的原因
- Java 接口和抽象类区别
- eclipse debug java源码时时总是显示 source not found
- Spring整合Tiles
- 使用ical4j和javamail发送会议邀请
- Eclipse Class Decompiler——Java反编译插件
- 怒学Java8系列一:Lambda表达式
- 比较器报错:Comparison method violates its general contract
- synchronized,当作用于方法与对象的不同之处
- Intellij Idea 的maven 项目会在 右键菜单 maven reimport 变成jdk 1.5
- Java极度性能调整
- spring AOP面向切面编程的四种实现方式
- java.lang.ClassNotFoundException: org.springframework.web.context.ContextLoaderListener
- spring 依赖注入的四种方式
- mac 下打开多个Eclipse
- jni&nbsp;java&nbsp;String&nbsp;转C&nbsp;char*工具类