spring整合mybatis
需要用到的jar
mybatis
org.mybatis mybatis 3.5.2 mysql-connector
mysql mysql-connector-java 5.1.6 spring
org.springframework spring-jdbc 5.2.0.RELEASE org.springframework spring-webmvc 5.2.0.RELEASE aspectjweaver
org.aspectj aspectjweaver 1.9.4 mybatis-spring(这里的重点)
org.mybatis mybatis-spring 2.0.2 maven静态资源过滤问题
src/main/java **/*.properties **/*.xml true
回顾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 listUser();
}
代理接口映射文件
UserMapper.xml
mybatis核心配置文件
mybatis
测试类
//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 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
知识基础
在开始使用 MyBatis-Spring 之前,你需要先熟悉 Spring 和 MyBatis 这两个框架和有关它们的术语。这很重要
MyBatis-Spring 需要以下版本:
如果使用 Maven 作为构建工具,仅需要在 pom.xml 中加入以下代码即可:
org.mybatis
mybatis-spring
2.0.2
SqlSessionFactoryBean与SqlSessionTemplate
要和 Spring 一起使用 MyBatis,需要在 Spring 应用上下文中定义至少两样东西:一个 SqlSessionFactory
和至少一个数据映射器类。
在 MyBatis-Spring 中,可使用 SqlSessionFactoryBean
来创建 SqlSessionFactory
。 要配置这个工厂 bean,只需要把下面代码放在 Spring 的 XML 配置文件中:
注意:SqlSessionFactory
需要一个 DataSource
(数据源)。 这可以是任意的 DataSource
,只需要和配置其它 Spring 数据库连接一样配置它就可以了。
在基础的 MyBatis 用法中,是通过 SqlSessionFactoryBuilder
来创建 SqlSessionFactory
的。 而在 MyBatis-Spring 中,则使用 SqlSessionFactoryBean
来创建。
在 MyBatis 中,你可以使用 SqlSessionFactory
来创建 SqlSession
。一旦你获得一个 session 之后,你可以使用它来执行映射了的语句,提交或回滚连接,最后,当不再需要它的时候,你可以关闭 session。
SqlSessionFactory
有一个唯一的必要属性:用于 JDBC 的 DataSource
。这可以是任意的 DataSource
对象,它的配置方法和其它 Spring 数据库连接是一样的。
一个常用的属性是 configLocation
,它用来指定 MyBatis 的 XML 配置文件路径。它在需要修改 MyBatis 的基础配置非常有用。通常,基础配置指的是
或
元素。
需要注意的是,这个配置文件并不需要是一个完整的 MyBatis 配置。确切地说,任何环境配置(
,数据源(
和 MyBatis 的事务管理器(
都会被忽略。SqlSessionFactoryBean 会创建它自有的 MyBatis 环境配置(Environment)
,并按要求设置自定义环境的值
SqlSessionTemplate
是 MyBatis-Spring 的核心。作为 SqlSession
的一个实现,这意味着可以使用它无缝代替你代码中已经在使用的 SqlSession
。
模板可以参与到 Spring 的事务管理中,并且由于其是线程安全的,可以供多个映射器类使用,你应该总是用 SqlSessionTemplate
来替换 MyBatis 默认的 DefaultSqlSession
实现。在同一应用程序中的不同类之间混杂使用可能会引起数据一致性的问题。
可以使用 SqlSessionFactory
作为构造方法的参数来创建 SqlSessionTemplate
对象。
现在,这个 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
:
整合实现
spring-config.xml
mybatis的配置文件,只保留别名(别名一样可以整合到spring的配置文件中)
spring-mybatis.xml
spring整合mybatis的配置文件
这里主要配置了
- 数据源datasource
- 会话工厂sqlSessionFactory
- 会话模板sqlSessionFactory
applicationContext.xml
spring的核心配置文件
在这里通过import将整合文件spring-mybatis.xml引入
UserMapperImp.java
增加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 listUser() {
UserMapper userMapper = sqlSessionTemplate.getMapper(UserMapper.class);
return userMapper.listUser();
}
}
注册bean实现
在applicationContext.xml中创建bean,因为spring主要就是IOC跟AOP
负责对象的管理和事务管理
测试
@Test
public void test2(){
ApplicationContext context = new ClassPathXmlApplicationContext("beans.xml");
UserMapper mapper = (UserMapper) context.getBean("userDao");
List 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 listUser() {
//getSqlSession()是SqlSessionDaoSupport中的方法
UserMapper userMapper = getSqlSession().getMapper(UserMapper.class);
return userMapper.listUser();
}
}
更便捷一点,可以将return写成一句话
return getSqlSession().getMapper(UserMapper.class).listUser();
注入bean
总结 : 整合到spring中以后可以完全不要mybatis的配置文件,除了这些方式可以实现整合之外,我们还可以使用注解来实现。