7.1Spring对MyBatis的整合思路
作为Bean容器,Spring框架提供了loC机制,可以接管所有组件的创建工作进行依赖管理,因而整合的主要工作就是把MyBatis框架使用中所涉及的核心组件配置到Spring容器中,交给Spring来创建和管理
具体来说,业务逻辑依赖基于MyBatis技术实现的DAO对象,核心是获取SqlSession实例.要获得SqlSession实例,则需要依赖SqlSessionFactory实例.而SqlSessionFactory是SqlSessionFactoryBuilder依据MyBatis配置文件中的数据源.SQL映射文件等信息来构建的
针对上述依赖关系,以往我们需要自行编码通过SqlSessionFactoryBuilder读取配置文件.构建SqlSessionFactory,进而获得SqlSession实例,满足业务逻辑对象对于数据访问的需要.随着Spring框架的引入,以上流程将全部移交给Spring,发挥Spring框架Bean容器的作用,接管组件的创建工作,管理组件的生命周期,并对组件之间的依赖关系进行解耦和管理
7.2Spring整合MyBatis的准备工作
1.在项目中加入Spring.MyBatis及整合相关的JAR文件
2.建立开发目录结构,创建实体类
3.创建数据访问接口
4.配置SQL映射文件
5.配置MyBatis配置文件
7.3实现Spring对MyBatis的整合
7.3.1配置数据源
对于任何持久化解决方案,数据库连接是首先要解决的问题.在Spring中,数据源作为一个重要的组件可以单独进行配置和维护
在Spring中配置数据源,首先要选择一种具体的数据源实现技术.目前流行的数据源实现有dbcp.c3p0.Proxool等,它们都实现了连接池功能.这里以配置dbcp数据源为例进行讲解,其他数据源的配置方法与类似,大家可以自行查阅相关资料进行学习.dbcp数据源隶属于Apache Commons项目.使用dbcp数据源,需要下载并在项目中添加commons-dbcp-1.4.jar和commons-pool-1.6.jar两个文件.
7.3.2配置SqlSessionFactoryBean
配置完数据源,就可以在此基础上集合SQL映射文件信息以及MyBatis配置文件中的其他信息创建SqlSessionFactory实例
在MyBatis中,SqlSessionFactory的实例需要使用SqlSessionFactoryBuilder创建;而在集成环境中用MyBatis-Spring整合包中的SqlSessionFactoryBean来代替.SqlSessionFactoryBean封装了使用SqlSessionFactoryBuilder创建SqlSessionFactory的过程,我们可以在Spring中以配置文件的形式,通过配置SqlSessionFactoryBean获得SqlSessionFactory实例
7.3.3使用SqlSessionTemplate实现数据库的操作
对于MyBatis而言,得到SqlSessionFactory实例,就可以进一步获取SqlSession实例进行数据库操作了.而在集成环境中,为了更好地使用SqlSession,充分利用Spring框架提供服务,MyBatis-Spring整合包提供了SqlSessionTemplate类.
SqlSessionTemplate类实现了MyBatis的SqlSession接口,可以替换MyBatis中原有的SqlSession实现类提供数据库访问操作.使用SqlSessionTemplate可以更好地与Spring服务融合并简化部分流程化的工作,可以保证和当前Spring事务关联,自动管理会话的生命周期,包括必要的关闭.提交和回滚操作
7.3.4编写业务逻辑代码并测试
7.3.5技能训练
7.4注入映射器实现
在上面的DAO实现中使用SqlSessionTemplate的方法,都是采用字符串来指定映射项,这种方式比较容易产生错误,如果存在拼写错误,在编译期无法识别,只能等到运行的时候才能发现.而且如果命名空间发生变化,会导致很多地方需要修改,不易维护。
MyBatis中可以使用SqlSession的getMapper (Class type)方法 .根据指定的映射器和映射文件直接生成实现类。这样不必自行编写映射器的实现类,就可以调用映射器的方法进行功能实现。
SqlSessionTemplate作为SqlSession 接口的实现,自然也具备相同作用的getMapper( )方法实现,但在集成环境中,直接在代码中使用getMapper( )方法并非最佳选择。利用MyBatis -Spring 提供的组件,可以不必每次调用getMappe( )方法,而是通过配置的方式直接为业务对象注入映射器实现,无需额外的编码。对于不包含其他非MyBatis的工作的数据访问操作.这是首选的做法。
7.4.1使用MapperFactoryBean注入映射器
如果仅使用SqlSessionTemplate执行基本数据访问操作,而不包含其他非MyBatis的工作,可以不必手工编码使用SqlSessionTemplate或者SqlSessionDaoSupport来实现此类DAO.MyBatis-Spring提供了MapperFactoryBean,能够以配置的方式生成映射器实现并注入给业务组件
注意 SQL映射文件中必须遵循以下命名原则:
(1)映射的命名空间和映射器接口的名称相同
(2)映射元素的id和映射接口的方法相同
7.4.2使用MapperFactoryTemplate实现数据库的操作
在Spring配置文件中使用MapperFactoryBean对映射器对配置.简化了DAO模块的编码,不过如果映射器很多,相应的配置也会很多,为了简化配置工作量,MyBatis-Sping中提供了MapperScannerConfigurer,它可以扫描指定包中的接口并将它们直接注册为MapperFactoryBean
7.4.3技能训练
7.5为业务层添加声明式事务
7.5.1配置声明式事务
业务层职能不仅仅是调用DAO这么简单,事务处理是如何企业级应用开发中不能回避的一个重要问题.以往我们通过在业务方法中硬编码的方式进行事务控制,这样做的弊端显而易见:事务代码分散在业务方法中难以重用,需要调整时工作量也比较大;复杂事务的编码难度较高,增加了开发难度等.Spring提供了声明事务处理机制,它基于AOP实现,无须编写任何事务管理代码,所以的工作全配置文件中完成.这意味着与业务代码分离,配置即可用,降低了开发和维护的难度
tx:method标签中的name属性是必需的,用于指定匹配的方法.这里需要对方法名进行约定,可用使用通配符(*).其他属性均可选,用于指定具体的事务,这些属性解释如下
➢propagation:事务传播机制.该属性可选的值有如下几种
◆REQUIRED:默认值,表示如果存在一个事务,则支持当前事务:如果当前没有事务则开启一个新事务
◆REQUIRES_NEW:表示总是开启一个新的事务.如果一个事务已经存在,则将这个存在的事务挂起,开启新事务执行该方法
◆MANDATORY:表示如果存在一个事务.则支持当前事务;如果当前事务的嵌套事务运行;如果当前事务,该取值与REQUIRED相同
◆NESTED:表示如果存在一个事务,则支持当前事务;如果当前没有事务,则将这个存在的事务挂起,然后执行该方法
◆NEVER:表示总是以非事务方式执行.如果当前存在一个活动的事务,则抛出异常
➢isolation:事务隔离等级.即当前事务和其他事务的隔离程度,在并发事务处理的情况下需要考虑它的设置.该属性可选的值有如下几种.
◆DEFAULT:默认值.表示使用数据库默认的事务隔离级别
◆READ_UUNCOMMITTED:未提交读
◆READ_COMMITTED:提交读
◆REPEATABLE_READ:可重复读
◆SERIALIZABLE:串行读
➢timeout:事务超时时间.允许事务云溪的最长时间,以秒为单位,超过给定的时间自动回滚,防止事务执行时间过长而影响系统性能.该属性需要底层的实现支持.默认值为01,表示不超过.
➢read-only:事务是否为只读,默认值为false.对于只执行查询功能的事务,把它设置为true,能提高事务处理的性能.
➢rollback-for:设定能够触发回滚的异常类型.Spring默认只在抛出Runtime Exception 时才标识提高事务回滚.可以通过全限定类名自行指定需要回滚事务的异常,多个类名用英文 逗号隔开
➢no-rollback-for:设定不触发回滚的异常类型.Spring默认checked Exception时才标识事务回滚.可以通过全限定类名自行指定吧需回滚的异常,多个类名用英文逗号隔开
7.5.2技能训练1
7.5.3使用注解实现声明式事务处理
Spring还支持使用注解配置声明式事务,所以用的注解Transaction
在业务实现类上添加@Transaction注解可为该类的所有业务统一添加事务处理.如果某一业务方法需要采用不同的事务规则,可以在业务方法上添加@Transactional注解单独进行设置.
@Transactional注解也可以设置事务属性的值,默认的@Transactional设置如下.
➢事务传播设置是PROPAGATION_REQUIRED.
➢事务隔离级别是ISOLATION_DEFAULT.
➢事务是读/写
➢事务超时默认是依赖事务系统的,或事务超时没有被支持
➢任何RuntimeException将触发事务回滚,但是任何checked Exception将不触发事务回滚.这些默认的设置当然是可以改变的.@Transactional注解的各属性如下