Mybatis执行流程、缓存原理及相关面试题汇总
作者:爪洼ing 发布时间:2022-10-15 23:30:28
一、Mybatis执行流程
具体分析Mybatis是如何操作数据库的!
1、定义我们的核心配置文件的路径,这个路径是从target/classes下开始找的!
String config = "mybatis-config.xml" ;
2、读取这个config表示的文件
InputStream inputStream= Resources.getResourceAsStream(config);
3、创建SqlSessionFactoryBuilder对象
SqlSessionFactoryBuilder builder = new SqlSessionFactoryBuilder();
4、创建SqlSessionFactory对象
SqlSessionFactory sessionFactory = builder.build(inputStream);
5、获取SqlSession对象,从SqlSessionFactory中获取
SqlSession sqlSession = sessionFactory.openSession();
6、【重点】执行我们的执行的的sql语句标识。sql文件中的namespace + sql标签的Id值
String sqlId = "com.sqx.dao.UserMapper.getUserById";
7、执行sql语句,通过sqlId找到语句,并执行!
List<Object> objects = sqlSession.selectList(sqlId);
8、关闭SqlSession
sqlSession.close();
总结
Myabtis启动的时候会加载mybatis-config.xml这个核心配置文件,得到要操作的数据库以及我们Mapper文件的的信息,然后将其以数据流的形式保存起来传给SqlSessionFactoryBuilder,创造出对应该数据库的SqlSessionFactory工厂,工厂生产SqlSession,这个SqlSession中包含了对该库的增删改查方法,我们需要通过namespace + sqlId
来确定我们执行的是哪个sql,这一步操作可以通过SqlSeesion.getMapper()拿到接口,通过调用接口的方法(方法绑定了我们的sql,包含namespace + sqlId
),执行对应的SQL语句!
二、Mybatis缓存
Mybatis缓存无论是一级缓存还是二级缓存都是本地缓存,都会占用JVM的内存,一旦Java停止缓存失效!
1、一级缓存
概述
一级缓存保存在我们的SqlSession当中
只要是同一个SqlSesisson(与数据库建立的会话)就会使用同一个缓存
一级缓存是我们Mybatis默认开启的!
如下是对一级缓存进行测试:
查看我们的测试结果
采用不同的SqlSession测试数据,进行如下修改!
我们再次查看测试结果
得出结论我们的一级缓存只是在同一个SqlSession当中有效!
2、二级缓存
概述
二级缓存又称"全局缓存",是基于namespace级别的缓存,一个namespace对应一个二级缓存!
工作机制:
一个会话查询一条数据,这个数据会被放在一级缓存当中
当我们会话关闭的时候,会把这个数据从1级缓存迁入二级缓存当中,新的会话就可以在二级缓存当中找到这个数据!
不同的会话查询不同的namespace的时候,会将不同namespace中的数据缓存到自己对应的缓存(map)中!
使用步骤 :
只需在需要使用缓存的namespace 中加入< cache/>
即可
<!--在当前Mapper.xml中使用二级缓存,并配置相关参数-->
<cache eviction="FIFO" flushInterval="60000" size="512" readOnly="true"/>
eviction :缓存的回收策略
LRU(最长时间未使用)、LFU(访问次数最少)、FIFO(默认)、SOFT(软引用)、WEAK(弱引用)
flushInterval : 缓存刷新间隔
缓存多久清空一次,默认是不会清空的,设置一个毫秒值
readOnly :是否只读
size :缓存存放多少元素
我们还可以通过配置实现自定义缓存
<!-- 开启mybatis的二级缓存,其本质还是我们mybatis帮我们实现了自身提供的缓存接口Cache,
也就是cache标签的type属性默认指定了缓存策略-->
<cache type=""/>
<!--本质上就是这种写法,我们可通该修改type来选择自定缓存策略-->
<cache type="rg.apache.ibatis.cache.impl.PerpetualCache"/>
3、缓存原理
如图所示
缓存执行流程
当我们的sql执行的时候,先去二级缓存namespace中查看是否存在缓存,
然后如果二级缓存不存在,查看当前sqlSession中一级缓存中是否存在,
最后一、二级缓存中都不存在的话那么就去数据库查询,
接着会将查询出来的结果保存在我们的一级缓存当中,
当前会话(SqlSession)结束,就会将一级缓存中的数据,同步到我们的二级缓存
三、相关面试题
1、为什么说 Mybatis 是半自动ORM映射工具?它与全自动的区别在哪里?
什么是ORM
ORM(Object Relational Mapping),对象关系映射,是一种为了解决关系型数据库数据与简单 Java 对象(POJO)建里映射关系的技术。
为什么说 Mybatis 是半自动ORM映射工具?它与全自动的区别在哪里?
首先,像 Hibernate、JPA 这种属于全自动 ORM 映射工具,使用 Hibernate 查询关联对象或者关联集合对象时,可以根据对象关系模型直接获取,所以它是全自动的。
而 Mybatis 在查询关联对象或关联集合对象时,需要手动编写 sql 来完成,所以,称之为半自动 ORM 映射工具。
换句话来解释就是说 MyBatis 是 半自动 ORM 最主要的一个原因是,它需要在 XML 或者注解里通过手动或插件生成 SQL,才能完成 SQL 执行结果与对象映射绑定。
2、简述一下Mybatis的执行流程吧 ?
MyBatis 是以一个 SqlSessionFactory 实例为核心,SqlSessionFactory 的实例可以通过 SqlSessionFactoryBuilder 获得。
SqlSessionFactoryBuilder 可以从 XML 配置文件或一个预先配置的 Configuration 实例来构建出 SqlSessionFactory 实例。
SqlSessionFactory 实例工厂可以生产 SqlSession ,它里面提供了在数据库执行 SQL 命令所需的所有方法。
3、说一下Mybatis的优缺点 ?
优点
基于SQL语句编写、相当灵活,SQL写在XML文件当中,解除了sql与程序代码的耦合,便于统一管理
消除了JDBC的冗余代码、能够与Spring很好的集成
缺点
SQL语句的编写工作量大,尤其是字段多,关联表多的情况下,对开发人员的SQL语句功底有一定的要求
SQL语句依赖数据库,导致数据库移植性差,不能随意更换数据库
4、#{} 和 ${} 的区别是什么 ?
#{} 是占位符,预编译处理,${}是拼接符,字符串替换,没有预编译处理
mybatis在处理#{}的时候,#{} 传入的参数是字符串,会将SQL中的#{} 替换为?调用PreparedStatement的Set方法赋值
myabtsi在处理的 时 候 , 就 是 原 值 传 入 , 就 是 把 {}的时候,就是原值传入,就是把的时候,就是原值传入,就是把{}替换为传来的值
#{}通过编译预处理可以有效地防止SQL注入问题,提高系统的安全性,${}则不能防止SQL注入
5、Mybatis的动态SQL有什么作用?执行原理是什么?有哪些常用标签?
mybatis动态sql可以在xml映射文件内,以标签的形式编写动态sql
执行原理:根据参数表达式的值完成逻辑判断,并且动态拼接sql
mybatis提供9种动态sql的标签:trim、where、set、foreach、if、choose、when、otherwise、bind
来源:https://blog.csdn.net/m0_46571920/article/details/122515795


猜你喜欢
- 对于使用avalonia的时候某些功能需要到一些提示,比如异常或者成功都需要对用户进行提示,所以需要单独实现弹窗功能,并且可以自定义内部组件
- 目前地图开放平台三大巨头:高德、百度、腾讯基本都支持Flutter插件开发集成。从这里也能看出Flutter的生态在逐渐的完善。下面介绍下在
- java中实现多线程的方法有两种:继承Thread类和实现runnable接口。1.继承Thread类,重写父类run()方法 public
- CountDownTimerCountDownTimer 是android 自带的一个倒计时类,使用这个类可以很简单的实现 倒计时功能Cou
- 一、在java中遍历一个文件夹里边的所有文件,可以有两种方式:1.递归遍历,通常也是开发者第一时间能想到的方法,递归遍历的优点是:实现起来相
- 这篇文章主要介绍了基于Java检查IPv6地址的合法性,文中通过示例代码介绍的非常详细,对大家的学习或者工作具有一定的参考学习价值,需要的朋
- 在开发应用过程中,客户端与服务端经常需要进行数据传输,涉及到重要隐私信息时,开发者自然会想到对其进行加密,即使传输过程中被“有心人”截取,也
- Java项目涉及到数据库交互,以往常用的是JDBC,现在则有Hibernate、Mybatis等这些持久化支持。项目中用到了MyBatis,
- 序列化(Serialize)是将对象转换成字节流,并将其用于存储或传输的过程,主要用途是保存对象的状态,以便在需要时重新创建该对象;反序列化
- 介绍Mybatis Generator(MBG)是Mybatis的一个代码生成工具。MBG解决了对数据库操作有最大影响的一些CRUD操作,很
- 本文实例讲述了Android开发实现popupWindow弹出窗口自定义布局与位置控制方法。分享给大家供大家参考,具体如下:布局文件:主布局
- 前言现如今几乎大多数Java应用,例如我们耳熟能详的tomcat, struts2, netty...等等数都数不过来的软件,要满足通用性,
- 先来看看效果:一、添加依赖库的步骤1.项目的gradle文件内的做以下改动allprojects { repositories
- 1.概述当Java 8发布时,其中最令人兴奋的特性之一就是 Lambda 表达式。Lambda 表达式是一种轻量级、简洁和高效的函数式编程方
- 需要自定义ListView。这里就交FloatDelListView吧。复写onTouchEvent方法。如下:@Override &nbs
- 本文,将介绍如何通过Java后端程序代码在PDF中创建工具提示。添加工具提示后,当鼠标悬停在页面上的元素时,将显示工具提示内容。导入jar包
- C++ 中二分查找递归非递归实现并分析二分查找在有序数列的查找过程中算法复杂度低,并且效率很高。因此较为受我们追捧。其实二分查找算法,是一个
- 关于logback日志的详解见这位仁兄的博客:Spring Boot-日志配置(超详细)我在这就开门见山直接介绍我们项目日志的配置使用吧!~
- 如今APP越来越多,我们每天所使用的的软件也越来越多,可是在我们不付费的情况下,App制造商如何实现,实现收入甚至是盈利呢?答案就是在我们打
- 效果和代码都非常直观:实例1:TimePicker<RelativeLayout xmlns:android="http:/