带你完虐面试官-mybatis篇

MyBatis执行流程

  1. 读取MyBatis配置文件:mybatis-config.xml加载运行环境和映射文件
  2. 构造会话工厂SqlSessionFactory
  3. 会话工厂创建SqlSession对象(包含了执行SQL语句的所有方法)
  4. 操作数据库的接口,Executor执行器,同时负责查询缓存的维护
  5. Executor接口的执行方法中有一个MappedStatement类型的参数,封装了映射信息
  6. 输入参数映射
  7. 输出结果映射

MyBatis延迟加载

MyBatis支持延迟加载,但默认没有开启

上面有两个表,一个是用户表,一个是订单表,是一对多的关系,也就是一个用户表对应多个订单表。

查询用户的时候,把用户所属的订单数据也查询出来,这个就是立即加载。

查询用户的时候,暂时不查询订单数据,当需要订单的时候,再查询订单,这个就是延迟加载。

延迟加载的原理

  1. 使用CGLIB创建目标对象的代理对象
  2. 当调用目标方法user.getOrderList()时,进入拦截器invoke方法,发现user.getOrderList()是NULL值,执行sql查询order列表
  3. 把order查询上来,然后调用user.serOrderList(List<Order> orderList),接着完成user.getOrderList()方法的调用

总结:

  • 延迟加载的意思是:就是在需要用到数据时才进行加载,不需要用到数据时就不加载数据。
  • MyBatis支持一对一关联对象和一对多关联集合对象的延迟加载
  • 在mybatis配置文件中,可以配置是否启用延迟加载lazyLoadingEnabled=true|false,默认是关闭的

延迟加载的底层原理

  1. 使用CGLIB创建目标对象的代理对象
  2. 当调用目标方法时,进入拦截器invoke方法,发现目标方法时null值,执行sql查询
  3. 获取数据以后,调用set方法设置属性值,再继续查询目标方法,就有值了

Mybatis的一级缓存和二级缓存

一般情况下,缓存的运用有以下两点:

  1. 请求命中缓存:将缓存的数据直接返回
  2. 请求未命中缓存:去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生效二级缓存 

注意事项:

  1. 对于缓存数据更新机制,当某一个作用域(一级缓存Session/二级缓存Namespaces)的进行了新增、修改、删除操作后,默认该作用域下所有select中的缓存将被clear
  2. 二级缓存需要缓存的数据实现Serializable接口
  3. 只有会话提交或者关闭以后,一级缓存中的数据才会转移到二级缓存中
  • 20
    点赞
  • 5
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值