MyBatis执行流程
- 读取MyBatis配置文件:mybatis-config.xml加载运行环境和映射文件
- 构造会话工厂SqlSessionFactory
- 会话工厂创建SqlSession对象(包含了执行SQL语句的所有方法)
- 操作数据库的接口,Executor执行器,同时负责查询缓存的维护
- Executor接口的执行方法中有一个MappedStatement类型的参数,封装了映射信息
- 输入参数映射
- 输出结果映射
MyBatis延迟加载
MyBatis支持延迟加载,但默认没有开启
上面有两个表,一个是用户表,一个是订单表,是一对多的关系,也就是一个用户表对应多个订单表。
查询用户的时候,把用户所属的订单数据也查询出来,这个就是立即加载。
查询用户的时候,暂时不查询订单数据,当需要订单的时候,再查询订单,这个就是延迟加载。
延迟加载的原理
- 使用CGLIB创建目标对象的代理对象
- 当调用目标方法user.getOrderList()时,进入拦截器invoke方法,发现user.getOrderList()是NULL值,执行sql查询order列表
- 把order查询上来,然后调用user.serOrderList(List<Order> orderList),接着完成user.getOrderList()方法的调用
总结:
- 延迟加载的意思是:就是在需要用到数据时才进行加载,不需要用到数据时就不加载数据。
- MyBatis支持一对一关联对象和一对多关联集合对象的延迟加载
- 在mybatis配置文件中,可以配置是否启用延迟加载lazyLoadingEnabled=true|false,默认是关闭的
延迟加载的底层原理
- 使用CGLIB创建目标对象的代理对象
- 当调用目标方法时,进入拦截器invoke方法,发现目标方法时null值,执行sql查询
- 获取数据以后,调用set方法设置属性值,再继续查询目标方法,就有值了
Mybatis的一级缓存和二级缓存
一般情况下,缓存的运用有以下两点:
- 请求命中缓存:将缓存的数据直接返回
- 请求未命中缓存:去DB查询数据,再将数据保存到缓存中,再返回
缓存的类别
- 本地缓存,基于PerpetualCache,本质是一个HashMap
- 一级缓存:作用域是session级别
- 二级缓存:作用域是namespace和mapper的作用域,不依赖于session
一级缓存
一级缓存:基于PerpetualCache的HashMap本地缓存,其存储作用域为Session,当Session进行flush或close之后,该Session中的所有Cache就将清空,默认打开一级缓存
//获取SqlSession对象,用它来执行sql
SqlSession sqlSession = sqlSessionFactory.openSession();
//获取UserMapper接口的代理对象
UserMapper userMapper1 = sqlSession.getMapper(UserMapper.class);
UserMapper userMapper2 = sqlSession.getMapper(UserMapper.class);
User user = userMapper1.selectById(6);
System.out.println(user);
User user1 = userMapper2.selectById(6);
System.out.println(user1);
sqlSession.close();
举个例子,以上代码是通过sqlSession工厂的openSession方法创建了一个SqlSession对象。在同一个sqlSession对象中,得到的两个userMapper共用的是同一个缓存,查询结果是同一个对象,只会执行一条sql查询。下面的代码是创建了两个sqlSession对象,分别用不同的sqlSession对象创建UserMapper,这次就会执行两条sql查询。
//获取SqlSession对象,用它来执行sql
SqlSession sqlSession1 = sqlSessionFactory.openSession();
//获取UserMapper接口的代理对象
UserMapper userMapper1 = sqlSession.getMapper(UserMapper.class);
User user1 = userMapper1.selectById(6);
System.out.println(user1);
sqlSession1.close();
SqlSession sqlSession2 = sqlSessionFactory.openSession();
UserMapper userMapper2 = sqlSession2.getMapper(UserMapper.class);
User user2 = userMapper2.selectById(6);
System.out.println(user2);
sqlSession2.close();
二级缓存
二级缓存是基于nameSpace和mapper的作用域起作用的,不是依赖于SqlSession,默认也是采用PerpetualCache,HashMap存储
二级缓存默认是关闭的
开启方式,两步:
1、全局配置文件
<settings>
<setting name="cacheEnabled" value="true"/>
</setting>
2、映射文件
使用<cache/>标签让当前mapper生效二级缓存
注意事项:
- 对于缓存数据更新机制,当某一个作用域(一级缓存Session/二级缓存Namespaces)的进行了新增、修改、删除操作后,默认该作用域下所有select中的缓存将被clear
- 二级缓存需要缓存的数据实现Serializable接口
- 只有会话提交或者关闭以后,一级缓存中的数据才会转移到二级缓存中