spring整合mybatis
需要用到的jar
-
mybatis
<dependency> <groupId>org.mybatis</groupId> <artifactId>mybatis</artifactId> <version>3.5.2</version> </dependency>
-
mysql-connector
<dependency> <groupId>mysql</groupId> <artifactId>mysql-connector-java</artifactId> <version>5.1.6</version> </dependency>
-
spring
<dependency> <groupId>org.springframework</groupId> <artifactId>spring-jdbc</artifactId> <version>5.2.0.RELEASE</version> </dependency> <dependency> <groupId>org.springframework</groupId> <artifactId>spring-webmvc</artifactId> <version>5.2.0.RELEASE</version> </dependency>
-
aspectjweaver
<dependency> <groupId>org.aspectj</groupId> <artifactId>aspectjweaver</artifactId> <version>1.9.4</version> </dependency>
-
mybatis-spring(这里的重点)
<dependency> <groupId>org.mybatis</groupId> <artifactId>mybatis-spring</artifactId> <version>2.0.2</version> </dependency>
-
maven静态资源过滤问题
<build> <resources> <resource> <directory>src/main/java</directory> <includes> <include>**/*.properties</include> <include>**/*.xml</include> </includes> <filtering>true</filtering> </resource> </resources> </build>
回顾mybatis
mysql创建表
CREATE DATABASE `mybatis`;
USE `mybatis`;
CREATE TABLE `user` (
`id` int(20) auto_increment PRIMARY KEY COMMENT '用户id',
`name` varchar(30) DEFAULT NULL COMMENT '姓名',
`pwd` varchar(30) DEFAULT NULL COMMENT '密码'
) ENGINE=InnoDB DEFAULT CHARSET=utf8;
实体类
import lombok.Data;//lombok插件
@Data
public class User {
private int id;
private String name;
private String pwd;
}
代理接口
public interface UserMapper {
List<User> listUser();
}
代理接口映射文件
UserMapper.xml
<?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.cong.dao.AccountMapper">
<select id="listUser" resultType="account">
select * from mybatis.account;
</select>
</mapper>
mybatis核心配置文件
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>
<typeAliases>
<package name="com.cong.pojo"/>
</typeAliases>
<environments default="mysql">
<environment id="mysql">
<transactionManager type="JDBC"></transactionManager>
<dataSource type="POOLED">
<property name="driver" value="com.mysql.jdbc.Driver"/>
<property name="url" value="jdbc:mysql://localhost:3306/mybatis?useSSL=true&useUnicode=true&characterEncoding=utf8"/>
<property name="username" value="root"/>
<property name="password" value="123456"/>
</dataSource>
</environment>
</environments>
<mappers>
<mapper class="com.cong.dao.AccountMapper"/>
</mappers>
</configuration>
测试类
//mybatis
@Test
public void test1() throws IOException {
InputStream inputStream = Resources.getResourceAsStream("mybatis-config.xml");
SqlSessionFactory factory = new SqlSessionFactoryBuilder().build(inputStream);
SqlSession sqlSession = factory.openSession();
UserMapper userMapper = sqlSession.getMapper(UserMapper.class);
List<User> userList = userMapper.listUser();
for (User user : userList) {
System.out.println(user);
}
}
总结一下,就是代理实现的过程,大概如下
DeaultSqlSession.getMapper() ->Configuration.getMapper() ->MapperRegistry.getMapper()->
MapperProxyFactory.newInstance(sqlSession)
到了这里明白了一点,也就是将通过MapperProxyFactory工厂生成了MapperProxy的代理类
进入MapperProxyFactory
MapperProxyFactory.newInstance(sqlSession) -> MapperProxyFactory.newInstance(mapperProxy)
至此,已经得到需要的代理类
通过MapperProxyFactory进入到MapperProxy类
发现MapperProxy实现了接口InvocationHandler,也就是动态代理
重点关注invoke()方法,看到在invoke方法里先获取MapperMethod类,然后调用mapperMethod.execute()
移到excute(),发现就是增删改查。
再看整个MapperMethod类
MapperMethod类是整个代理机制的核心类,对SqlSession中的操作进行了封装使用。 该类里有两个内部类SqlCommand和MethodSignature。 SqlCommand用来封装CRUD操作,也就是我们在xml中配置的操作的节点。每个节点都会生成一个MappedStatement类。MethodSignature用来封装方法的参数以及返回类型,在execute的方法中我们发现在这里又回到了SqlSession中的接口调用,和我们自己实现UerDao接口的方式中直接用SqlSession对象调用DefaultSqlSession的实现类的方法是一样的,经过一大圈的代理又回到了原地,这就是整个动态代理的实现过程了。
spring整合mybatis
什么是 MyBatis-Spring?
MyBatis-Spring 会帮助你将 MyBatis 代码无缝地整合到 Spring 中。首先了解一下spring-mybatis这个jar包
知识基础
在开始使用 MyBatis-Spring 之前,你需要先熟悉 Spring 和 MyBatis 这两个框架和有关它们的术语。这很重要
MyBatis-Spring 需要以下版本:
MyBatis-Spring | MyBatis | Spring 框架 | Spring Batch | Java |
---|---|---|---|---|
2.0 | 3.5+ | 5.0+ | 4.0+ | Java 8+ |
1.3 | 3.4+ | 3.2.2+ | 2.1+ | Java 6+ |
如果使用 Maven 作为构建工具,仅需要在 pom.xml 中加入以下代码即可:
<dependency>
<groupId>org.mybatis</groupId>
<artifactId>mybatis-spring</artifactId>
<version>2.0.2</version>
</dependency>
SqlSessionFactoryBean与SqlSessionTemplate
要和 Spring 一起使用 MyBatis,需要在 Spring 应用上下文中定义至少两样东西:一个 SqlSessionFactory
和至少一个数据映射器类。
在 MyBatis-Spring 中,可使用 SqlSessionFactoryBean
来创建 SqlSessionFactory
。 要配置这个工厂 bean,只需要把下面代码放在 Spring 的 XML 配置文件中:
<bean id="sqlSessionFactory" class="org.mybatis.spring.SqlSessionFactoryBean">
<property name="dataSource" ref="dataSource" />
</bean>
注意:SqlSessionFactory
需要一个 DataSource
(数据源)。 这可以是任意的 DataSource
,只需要和配置其它 Spring 数据库连接一样配置它就可以了。
在基础的 MyBatis 用法中,是通过 SqlSessionFactoryBuilder
来创建 SqlSessionFactory
的。 而在 MyBatis-Spring 中,则使用 SqlSessionFactoryBean
来创建。
在 MyBatis 中,你可以使用 SqlSessionFactory
来创建 SqlSession
。一旦你获得一个 session 之后,你可以使用它来执行映射了的语句,提交或回滚连接,最后,当不再需要它的时候,你可以关闭 session。
SqlSessionFactory
有一个唯一的必要属性:用于 JDBC 的 DataSource
。这可以是任意的 DataSource
对象,它的配置方法和其它 Spring 数据库连接是一样的。
一个常用的属性是 configLocation
,它用来指定 MyBatis 的 XML 配置文件路径。它在需要修改 MyBatis 的基础配置非常有用。通常,基础配置指的是<settings>
或<typeAliases>
元素。
需要注意的是,这个配置文件并不需要是一个完整的 MyBatis 配置。确切地说,任何环境配置(<environments>)
,数据源(<DataSource>)
和 MyBatis 的事务管理器(<transactionManager>)
都会被忽略。SqlSessionFactoryBean 会创建它自有的 MyBatis 环境配置(Environment)
,并按要求设置自定义环境的值
SqlSessionTemplate
是 MyBatis-Spring 的核心。作为 SqlSession
的一个实现,这意味着可以使用它无缝代替你代码中已经在使用的 SqlSession
。
模板可以参与到 Spring 的事务管理中,并且由于其是线程安全的,可以供多个映射器类使用,你应该总是用 SqlSessionTemplate
来替换 MyBatis 默认的 DefaultSqlSession
实现。在同一应用程序中的不同类之间混杂使用可能会引起数据一致性的问题。
可以使用 SqlSessionFactory
作为构造方法的参数来创建 SqlSessionTemplate
对象。
<bean id="sqlSession" class="org.mybatis.spring.SqlSessionTemplate">
<constructor-arg index="0" ref="sqlSessionFactory" />
</bean>
现在,这个 bean 就可以直接注入到你的 DAO bean 中了。你需要在你的 bean 中添加一个 SqlSession 属性,就像下面这样:
public class UserDaoImpl implements UserDao {
private SqlSession sqlSession;
public void setSqlSession(SqlSession sqlSession) {
this.sqlSession = sqlSession;
}
public User getUser(String userId) {
return sqlSession.getMapper...;
}
}
按下面这样,注入 SqlSessionTemplate
:
<bean id="userDao" class="org.mybatis.spring.sample.dao.UserDaoImpl">
<property name="sqlSession" ref="sqlSession" />
</bean>
整合实现
User与UserMapper.xml与上面的一致。
UserMapperImp.java
上面讲到,SqlSessionTemplate
是 MyBatis-Spring 的核心。作为 SqlSession
的一个实现,这意味着可以使用它无缝代替你代码中已经在使用的 SqlSession
。
增加mapper接口的实现类,私有化sqlSessionTemplate,提供setter方法,以供spring注入到容器中
public class UserMapperImpl implements UserMapper{
private SqlSessionTemplate sqlSessionTemplate;//sqlSession不用我们自己创建了,Spring来管理
public void setSqlSessionTemplate(SqlSessionTemplate sqlSessionTemplate) {
this.sqlSessionTemplate = sqlSessionTemplate;
}
@Override
public List<User> listUser() {
UserMapper userMapper = sqlSessionTemplate.getMapper(UserMapper.class);
return userMapper.listUser();
}
}
spring-config.xml
mybatis的配置文件,只保留别名(别名一样可以整合到spring的配置文件中)
<?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>
<typeAliases>
<package name="com.cong.pojo"/>
</typeAliases>
</configuration>
spring-mybatis.xml
spring整合mybatis的配置文件
这里主要配置了
- 数据源datasource
- 会话工厂sqlSessionFactory
- 会话模板sqlSessionFactory
- 注册UserMapper实现类
<?xml version="1.0" encoding="UTF-8"?>
<beans xmlns="http://www.springframework.org/schema/beans"
xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"
xsi:schemaLocation="http://www.springframework.org/schema/beans
https://www.springframework.org/schema/beans/spring-beans.xsd">
<!--datasource-->
<!--配置数据源:数据源有非常多,可以使用第三方的,这里使用Spring的-->
<bean id="dataSources" class="org.springframework.jdbc.datasource.DriverManagerDataSource">
<property name="driverClassName" value="com.mysql.jdbc.Driver"/>
<property name="url" value="jdbc:mysql://localhost:3306/mybatis?useSSL=true&useUnicode=true&characterEncoding=utf8"/>
<property name="username" value="root"/>
<property name="password" value="123456"/>
</bean>
<!--sqlSessionFactory-->
<bean id="sqlSessionFactory" class="org.mybatis.spring.SqlSessionFactoryBean">
<property name="dataSource" ref="dataSources"/>
<!--关联Mybatis-->
<property name="configLocation" value="mybatis-config.xml"/>
<!--映射文件-->
<property name="mapperLocations" value="classpath:com/cong/dao/*Mapper.xml"/>
<!--别名-->
<!--<property name="typeAliases" value="com.cong.pojo.Account"/>-->
</bean>
<!--SqlSessionTemplate-->
<bean id="sqlSession" class="org.mybatis.spring.SqlSessionTemplate">
<!--利用构造器注入,没有setter-->
<constructor-arg index="0" ref="sqlSessionFactory"/>
</bean>
<!--注册UserMapper实现类-->
<bean id="userMapper" class="com.cong.dao.UserMapperImpl">
<property name="sqlSessionTemplate" ref="sqlSession"/>
</bean>
</beans>
applicationContext.xml
spring的核心配置文件
在这里通过import将整合文件spring-mybatis.xml引入
<?xml version="1.0" encoding="UTF-8"?>
<beans xmlns="http://www.springframework.org/schema/beans"
xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"
xsi:schemaLocation="http://www.springframework.org/schema/beans
https://www.springframework.org/schema/beans/spring-beans.xsd">
<!--引入整合的配置文件-->
<import resource="classpath:spring-dao.xml"/>
</beans>
测试
@Test
public void test2(){
ApplicationContext context = new ClassPathXmlApplicationContext("beans.xml");
UserMapper mapper = (UserMapper) context.getBean("userDao");
List<User> user = mapper.selectUser();
System.out.println(user);
}
结果成功输出!
User(id=1, name=cong, pwd=ajd123.)
User(id=2, name=rainbow, pwd=182184)
改进
mybatis-spring1.2.3版以上的才有这个 .
dao继承Support类 , 直接利用 getSqlSession() 获得 , 然后直接注入SqlSessionFactory . 比起方式1 , 不需要管理SqlSessionTemplate , 而且对事务的支持更加友好 .
再写一个UserMapper的实现类UserMapperImpl2
通过继承SqlSessionDaoSupport来消除SqlSessionTemplate的管理
//可以通过继承SqlSessionDaoSupport来消除SqlSessionTemplate的显示注入
public class UserMapperImpl2 extends SqlSessionDaoSupport implements UserMapper{
@Override
public List<User> listUser() {
//getSqlSession()是SqlSessionDaoSupport中的方法
UserMapper userMapper = getSqlSession().getMapper(UserMapper.class);
return userMapper.listUser();
}
}
更便捷一点,可以将return写成一句话
return getSqlSession().getMapper(UserMapper.class).listUser();
注入bean
<bean id="userMapper2" class="com.cong.dao.UserMapperImpl2">
<property name="sqlSessionTemplate" ref="sqlSession"/>
</bean>
总结 : 整合到spring中以后可以完全不要mybatis的配置文件,除了这些方式可以实现整合之外,我们还可以使用注解来实现。
再改进
回顾一下mybatis,我们的代码中根本没有映射接口的实现类。
在spring整合mybatis中也可以做到,使用MapperFactoryBean
。
如果映射器接口 UserMapper 在相同的类路径下有对应的 MyBatis XML 映射器配置文件,将会被 MapperFactoryBean
自动解析。不需要在 MyBatis 配置文件中显式配置映射器,除非映射器配置文件与接口类不在同一个类路径下。参考 SqlSessionFactoryBean
的 configLocation
属性以获取更多信息。
注意 MapperFactoryBean
需要配置一个 SqlSessionFactory
或 SqlSessionTemplate
。它们可以分别通过 sqlSessionFactory
和 sqlSessionTemplate
属性来进行设置。如果两者都被设置,SqlSessionFactory
将被忽略。由于 SqlSessionTemplate
已经设置了一个 session 工厂,MapperFactoryBean
将使用那个工厂。
mybatis-dao2.xml
<?xml version="1.0" encoding="UTF-8"?>
<beans xmlns="http://www.springframework.org/schema/beans"
xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"
xsi:schemaLocation="http://www.springframework.org/schema/beans
https://www.springframework.org/schema/beans/spring-beans.xsd">
<bean id="dataSources" class="org.springframework.jdbc.datasource.DriverManagerDataSource">
<property name="driverClassName" value="com.mysql.jdbc.Driver"/>
<property name="url" value="jdbc:mysql://localhost:3306/mybatis?useSSL=true&useUnicode=true&characterEncoding=utf8"/>
<property name="username" value="root"/>
<property name="password" value="123456"/>
</bean>
<bean id="sqlSessionFactory" class="org.mybatis.spring.SqlSessionFactoryBean">
<property name="dataSource" ref="dataSources"/>
<property name="configLocation" value="mybatis-config.xml"/>
<property name="mapperLocations" value="classpath:com/cong/dao/*Mapper.xml"/>
</bean>
<!-- 注册userMapper -->
<bean id="userMapper" class="org.mybatis.spring.mapper.MapperFactoryBean">
<property name="mapperInterface" value="com.cong.dao.UserMapper"/>
<property name="sqlSessionFactory" ref="sqlSessionFactory"/>
</bean>
</beans>
applicationContext2.xml
<?xml version="1.0" encoding="UTF-8"?>
<beans xmlns="http://www.springframework.org/schema/beans"
xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"
xsi:schemaLocation="http://www.springframework.org/schema/beans
https://www.springframework.org/schema/beans/spring-beans.xsd">
<!--引入整合的配置文件-->
<import resource="classpath:spring-dao2.xml"/>
</beans>
测试
@Test
public void test4(){
ApplicationContext context = new ClassPathXmlApplicationContext("applicationContext2.xml");
UserMapper userMapper = (UserMapper) context.getBean("userMapper");
List<User> userList = userMapper.listUser();
for (User user : userList) {
System.out.println(user.toString());
}
}
最后,扫描mapper接口
如果,我们有很多个mapper接口,一个一个配置的话,是很麻烦的,可以通过扫描一个包,实现多个mapper的注入。
用
<!--或者使用spring的扫描mapper-->
<bean class="org.mybatis.spring.mapper.MapperScannerConfigurer">
<property name="basePackage" value="com.cong.dao"/>
<!--指定sqlSessionFactory或sqlSessionTemplate可不设置-->
<!--<property name="sqlSessionFactoryBeanName" value="sqlSessionFactory"/>-->
<property name="sqlSessionTemplateBeanName" value="sqlSessionTemplate"/>
</bean>
<bean id="sqlSessionTemplate" class="org.mybatis.spring.SqlSessionTemplate">
<!--利用构造器注入,没有setter-->
<constructor-arg index="0" ref="sqlSessionFactory"/>
</bean>
可以替代上面的
<!-- 注册userMapper -->
<bean id="userMapper" class="org.mybatis.spring.mapper.MapperFactoryBean">
<property name="mapperInterface" value="com.cong.dao.UserMapper"/>
<property name="sqlSessionFactory" ref="sqlSessionFactory"/>
</bean>
需要说明的是
如果你需要指定 sqlSessionFactory
或 sqlSessionTemplate
,那你应该要指定的是 bean 名而不是 bean 的引用,因此要使用 value
属性而不是通常的 ref
属性。
提示 在 MyBatis-Spring 1.0.2 之前,sqlSessionFactoryBean
和 sqlSessionTemplateBean
属性是唯一可用的属性。但由于 MapperScannerConfigurer
在启动过程中比 PropertyPlaceholderConfigurer
运行得更早,经常会产生错误。基于这个原因,上述的属性已被废弃,现在建议使用 sqlSessionFactoryBeanName
和 sqlSessionTemplateBeanName
属性。