通过示例讲解MyBatis和Spring整合的流程。
前言
这个MyBatis和Spring整合示例,是我今年3月份刚转岗时需要熟悉MyBatis时写的,主要是想知道项目是如何从0到1,在Spring中用到MyBatis。这个示例是参考“C语言中文网”,但是里面的示例不能直接运行,可能是因为版本等原因,当时为了集成MyBatis和Spring,倒腾了好几天,现在回过来看,如果你完全不了解MyBatis的基础知识,直接照葫芦画瓢,虽然也能解决问题,但是排查问题的过程其实是非常痛苦的,下面我将给一个两者集成的完整示例,让大家少走弯路。
项目准备
DB使用的是Mysql,pom.xml需要添加的依赖包:
<dependency> <groupId>org.mybatis</groupId> <artifactId>mybatis</artifactId> <version>3.4.6</version> </dependency> <dependency> <groupId>org.mybatis</groupId> <artifactId>mybatis-spring</artifactId> <version>2.0.0</version> </dependency> <dependency> <groupId>mysql</groupId> <artifactId>mysql-connector-java</artifactId> <version>6.0.6</version> </dependency>
DB结构:
CREATE TABLE `user_test` ( `uid` tinyint(2) NOT NULL, `uname` varchar(20) DEFAULT NULL, `usex` varchar(10) DEFAULT NULL, PRIMARY KEY (`uid`) ) ENGINE=InnoDB DEFAULT CHARSET=utf8
DB初始数据:
uid | uname | usex |
1 | 张三 | 女 |
2 | 陈恒 | 男 |
3 | 楼仔 | 男 |
集成示例
第1步:创建持久化类
@Data public class MyUser { private Integer uid; // 主键 private String uname; private String usex; }
第2步:创建映射文件
<?xml version="1.0" encoding="UTF-8"?> <!DOCTYPE mapper PUBLIC "-//mybatis.org//DTD Mapper 3.0//EN" "http://mybatis.org/dtd/mybatis-3-mapper.dtd"> <mapper namespace="com.mybatis.dao.UserDao"> <!-- 根据uid查询一个用户信息 --> <select id="selectUserById" parameterType="Integer" resultType="com.mybatis.entity.MyUser"> select * from user_test where uid = #{uid} </select> <!-- 查询所有用户信息 --> <select id="selectAllUser" resultType="com.mybatis.entity.MyUser"> select * from user_test </select> <!-- 添加一个用户,#{uname}为 com.mybatis.po.MyUser 的属性值 --> <insert id="addUser" parameterType="com.mybatis.entity.MyUser"> insert into user_test (uid,uname,usex) values(#{uid},#{uname},#{usex}) </insert> <!--修改一个用户 --> <update id="updateUser" parameterType="com.mybatis.entity.MyUser"> update user_test set uname =#{uname},usex = #{usex} where uid = #{uid} </update> <!-- 删除一个用户 --> <delete id="deleteUser" parameterType="Integer"> delete from user_test where uid= #{uid} </delete> </mapper>
第3步:创建映射文件对应的接口
@Repository("userDao") @Mapper /* * 使用Spring自动扫描MyBatis的接口并装配 (Spring将指定包中所有被@Mapper注解标注的接口自动装配为MyBatis的映射接口 */ public interface UserDao { /** * 接口方法对应的SQL映射文件中的id */ public MyUser selectUserById(Integer uid); public List<MyUser> selectAllUser(); public int addUser(MyUser user); public int updateUser(MyUser user); public int deleteUser(Integer uid); }
之前直接使用MyBatis时,是没有这个接口文件的,添加这个文件,是为了能让Spring和Mybatis的操作通过该文件关联起来,即通过上述提供的接口实现对DB的操作。
第4步:新建mybatis配置
<?xml version="1.0" encoding="utf-8"?> <!DOCTYPE configuration PUBLIC "-//mybatis.org//DTD Config 3.0//EN" "http://mybatis.org/dtd/mybatis-3-config.dtd"> <configuration> <settings> <setting name="logImpl" value="LOG4J" /> <setting name="cacheEnabled" value="false"/> <setting name="defaultExecutorType" value="REUSE"/> <setting name="useGeneratedKeys" value="true"/> </settings> </configuration>
由于后面会将两者整合,所以关于MyBatis中environments和mappers的配置,将会全部移到applicationContext.xml中,这个是直接用MyBatis中一个很大的区别。
第5步:新增DB配置文件
新增DB配置文件jdbc.properties,用于保存DB配置信息:
jdbc.shop.url=jdbc:mysql://xxx:3104/xm_jointly?characterEncoding=utf8 jdbc.shop.username=jointly_xx jdbc.shop.password=G-uTlU-xxx
第6步:增加配置信息
需要在applicationContext.xml中新增配置信息,这里就是整个集成的关键所在:
<!-- 用于加载配置文件 --> <bean class="org.springframework.beans.factory.config.PropertyPlaceholderConfigurer"> <property name="location"> <value>classpath:com/mybatis/config/datasources/jdbc.properties</value> </property> </bean> <!-- 定义数据源 --> <bean id="dataSource" class="org.springframework.jdbc.datasource.DriverManagerDataSource"> <property name="url" value="${jdbc.shop.url}"></property> <property name="username" value="${jdbc.shop.username}"></property> <property name="password" value="${jdbc.shop.password}"></property> </bean> <!-- 添加事务支持 --> <bean id="txManager" class="org.springframework.jdbc.datasource.DataSourceTransactionManager"> <property name="dataSource" ref="dataSource" /> </bean> <!-- 注册事务管理驱动 --> <tx:annotation-driven transaction-manager="txManager" /> <!-- 配置SqlSessionFactoryBean --> <bean id="sqlSessionFactory" class="org.mybatis.spring.SqlSessionFactoryBean"> <!-- 引用数据源组件 --> <property name="dataSource" ref="dataSource" /> <!-- 引用DB XML配置文件 --> <property name="mapperLocations" value="classpath*:com/mybatis/mapper/UserMapper.xml" /> <!-- 引用MyBatis配置文件中的配置 --> <property name="configLocation" value="classpath:com/mybatis/config/datasources/mybatis-config.xml" /> </bean> <!-- Mapper代理开发,使用Spring自动扫描MyBatis的接口并装配 (Spring将指定包中的所有被@Mapper注解标注的接口自动装配为MyBatis的映射接口) --> <bean class="org.mybatis.spring.mapper.MapperScannerConfigurer"> <!-- mybatis-spring组件的扫描器,com.dao只需要接口(接口方法与SQL映射文件中的相同) --> <property name="basePackage" value="com.mybatis.dao" /> <property name="sqlSessionFactoryBeanName" value="sqlSessionFactory" /> </bean>
这个配置需要解读一下:
- 用于加载配置文件:用于存放DB的配置信息,就是上文中提到的jdbc.properties配置文件,这样的好处就是不需要将DB配置信息在XML中写死。
- 定义数据源:这个就是DB的账户和密码了,jdbc.shop.url和jdbc.shop.username等就是DB的配置信息,保存在jdbc.properties配置文件中。
- 添加事务支持:之前这个事务支持的配置是在MyBatis中的environments中,现在给挪到了这里,还是采用JDBC的方式。注意一下,添加事务一定需要注册事务管理驱动。
- 配置SqlSessionFactoryBean:这个其实还是MyBatis中之前的配置,dataSource是数据源、mapperLocations是XML映射文件、configLocation是MyBatis的原始配置,相当于把之前未集成的MyBatis配置拆分了一下,把里面的environments和mappers给挪到这里来了。这个其实就是用来构建sqlSessionFactory会话工厂,我们可以再回顾一下MyBatis的调用流程:
这个建议大家看看文章《【MyBatis系列1】基础知识(下)》,应该就能很明显看出他们的区别,所以为什么需要大家掌握MyBatis未集成的实现方式,这样可以知道它的原貌,包括后面继承Spring Boost,原理都一样,只是换了一种展示方式。
- Spring自动扫描MyBatis的接口并装配:这里就是两者结合的核心,怎么结合呢?通过com.mybatis.dao.UserDao接口,就是我们“第3步:创建映射文件对应的接口”,所以这个才是两者集成的核心,里面的basePackage指定接口路径,sqlSessionFactoryBeanName是用到的sqlSessionFactory会话工厂。
到这里,是不是就很清晰了呢,我可是结合MyBatis未集成的方式进行对比讲解,这样就知道Spring对于两者的集成都做了哪些事情。
第7步:测试示例
我们先常见一个测试类:
@Controller("userController") public class UserController { @Autowired private UserDao userDao; public void test() { // 查询一个用户 MyUser auser = userDao.selectUserById(1); System.out.println(auser); System.out.println("============================"); // 查询所有用户 List<MyUser> list = userDao.selectAllUser(); for (MyUser myUser : list) { System.out.println(myUser); } } }
测试示例:
// 用于测试MyBatis和Spring的集成 public class SpringMyBatisTest { public static void main(String[] args) { String xmlPath = "applicationContext.xml"; ApplicationContext applicationContext = new ClassPathXmlApplicationContext(xmlPath); UserController uc = (UserController) applicationContext.getBean("userController"); uc.test(); } }
输出结果:
MyUser(uid=1, uname=张三, usex=女) MyUser(uid=2, uname=陈恒, usex=男) MyUser(uid=3, uname=楼仔, usex=男)
项目框架
如果只根据上面的示例想让项目跑起来,对于小白来说还是有点困难的,因为你不知道哪个文件放哪里,这里我直接给出项目视图,是不是感觉楼哥超级暖呢。
后记
学习某个工具,感觉还是需要从演变的源头学起,比如学习MyBatis,首先就去学习MyBatis和Spring Boost的集成,虽然知道怎么使用,但是出了问题就不知道怎么定位,所以还是需要掌握基本的流程和原理。也比如学习注解,最开始是从XML演变过来的,如果对XML不了解,直接去学习注解,那么理解的也不深刻。
MyBatis这个系列中,我会把MyBatis与Spring Boost的整合也写一下,等程序写完后,再出这篇文章哈!