MyBatis八股文

1、什么是MyBatis框架?

MyBatis 是一款优秀的持久层框架,它支持定制化 SQL、存储过程以及高级映射,它内部封装了jdbc,不需要我们再写JDBC连接、使开发者只需要关注sql语句本身和业务,而不需要花费精力去处理加载驱动、创建连接、创建statement等繁杂的过程。直接通过配置文件或maven驱动包的方式加载导入就行。

工作原理
在这里插入图片描述

  1. 读取 MyBatis 配置文件:mybatis-config.xml 为 MyBatis 的全局配置文件,配置了 MyBatis 的运行环境等信息,例如数据库连接信息。
  2. 加载映射文件。映射文件即 SQL 映射文件,该文件中配置了操作数据库的 SQL 语句,需要在 MyBatis 配置文件 mybatis-config.xml 中加载。mybatis-config.xml 文件可以加载多个映射文件,每个文件对应数据库中的一张表。
  3. 构造会话工厂:通过 MyBatis 的环境等配置信息构建会话工厂 SqlSessionFactory。
  4. 创建会话对象:由会话工厂创建 SqlSession 对象,该对象中包含了执行 SQL 语句的所有方法。
  5. Executor 执行器:MyBatis 底层定义了一个 Executor 接口来操作数据库,它将根据 SqlSession 传递的参数动态地生成需要执行的 SQL 语句,同时负责查询缓存的维护。
  6. MappedStatement 对象:在 Executor 接口的执行方法中有一个 MappedStatement 类型的参 数,该参数是对映射信息的封装,用于存储要映射的 SQL 语句的 id、参数等信息。
  7. 输入参数映射:输入参数类型可以是 Map、List 等集合类型,也可以是基本数据类型和 POJO 类 型。输入参数映射过程类似于 JDBC 对 preparedStatement 对象设置参数的过程。
  8. 输出结果映射:输出结果类型可以是 Map、 List 等集合类型,也可以是基本数据类型和 POJO 类 型。输出结果映射过程类似于 JDBC 对结果集的解析过程。
    2、MyBatis的优点有哪些?
    1)Mybatis基于SQL语句编程,很灵活,不会有任何影响现有应用程序或数据库的设计,SQL编写的XML,删除SQL和程序代码的耦合,便于统一管理,提供了XML标记,使您能够编写动态SQL语句和重用它们。
    2)与JDBC相比,它减少了50%以上的代码量,消除了大量冗余的JDBC代码,不需要手动切换连接;
    3)与各种数据库的良好兼容性(因为MyBatis使用JDBC连接数据库,所以只要JDBC支持数据库MyBatis就支持)。
    4)与Spring和Spring MVC框架的良好集成;
    5)提供映射标签,支持对象与数据库ORM字段关系映射;提供对象关系映射标签,支持对象关系组件的维护。

3、#{}和KaTeX parse error: Expected 'EOF', got '#' at position 13: {}的区别是什么? 1)#̲{}是预编译处理,{}是字符串替换
2)Mybatis 在处理#{}时,会将 sql 中的#{}替换为?号,调用 PreparedStatement 的 set 方法来赋值;在处理时,会把{}替换成变量的值。使用#{}可以有效的防止SQL注入,提高系统安全性。
大家都知道Mybatis 的Mapper.xml语句中parameterType向SQL语句传参有两种方式:#{}和KaTeX parse error: Expected 'EOF', got '#' at position 12: {} 我们经常使用的是#̲{},一般解说是因为这种方式可…{}在动态解析时候,会传入参数字符串,这种传参就会出现sql注入的安全性问题。
select * from student where student_name = “1’ OR ‘1’='1”;

4、当实体类中的属性名和表中的字段名不一样,怎么处理?

select * from employees where id = #{id}
<!-- 自定义高级映射 -->
<resultMap type="com.tt.mybatis.entities.Employee" id="myMap">
	<!-- 映射主键 -->
	<id column="id" property="id"/>
	<!-- 映射其他列 -->
	<result column="last_name" property="lastName"/>
	<result column="email" property="email"/>
	<result column="salary" property="salary"/>
	<result column="dept_id" property="deptId"/>
</resultMap>
第一种是使用标签,逐一定义数据库列名和对象属性名之间的映射关系。 第二种是使用sql列的别名功能,将列的别名书写为对象属性名。 有了列名与属性名的映射关系后,Mybatis通过反射创建对象,同时使用反射给对象的属性逐一赋值并返回,那些找不到映射关系的属性,是无法完成赋值的。

5、模糊查询like语句该怎么写?
1、 “%”#{question}“%” 注意:因为#{…}解析成sql语句时候,会在变量外侧自动加单引号’ ',所以 这里 % 需要使用双引号" “,不能使用单引号 ’ ',不然会查不到任何结果。

select * from foo where bar like “%”#{value}”%"

2、CONCAT(’%’,#{question},’%’) 使用CONCAT()函数,(推荐)
6、MyBatis Dao层接口的工作原理是什么?Dao接口里的方法,参数不同时,方法能重载吗?
举例说明: Dao接口就是Mapper接口,—可以基于注解的方式创建接口,接口内定义抽象方法;
public interface UserMapper {

@Select("select * from users where id=#{id}")
public User getUserById(int id);

}
接口的全限名,就是映射文件中的namespace的值:


select * from users where id=#{id}


Mapper接口没有实现类。当调用接口方法时,将接口名称+方法名称与字符串连接作为键值,以唯一定位MappedStatement。
在Mybatis中,每个select 、insert 、update、delete标记都被解析为一个MappedStatement对象。
Dao 接口即 Mapper 接口,接口的全限名,就是映射文件中的 namespace 的值;接口的方法名,就是映射文件中 Mapper 的 Statement 的 id 值;接口方法内的参数,就是传递给 sql 的参数。

关于重载和工作原理解释
1)Dao接口中的2个方法不能被覆盖,因为它是一个全名+方法名的保存和查找策略。
2)Dao接口的工作原理是JDK动态代理。mybatis会对每一个mapper代理生成一个mapperProxy对象,代理对象会拦截接口方法,转而自动对应到sqlsession上,最终由Executor执行。

7、Mybatis是如何进行分页的?
Mybatis 使用RowBounds对象进行分页,它是针对 ResultSet 结果集执行的内存分页,而非物理分页。可以在 sql 内直接书写带有物理分页的参数来完成物理分页功能,也可以使用分页插件来完成物理分页。下面看看Mybatis的如何进行分页。

8、MyBatis都有哪些Excutor执行器,它们之间的区别是什么?
Mybatis有三种基本的Executor执行器,SimpleExecutor、ReuseExecutor、BatchExecutor。
1)SimpleExecutor:每执行一次update或select,就开启一个Statement对象,用完立刻关闭Statement对象。
2)ReuseExecutor:执行update或select,以sql作为key查找Statement对象,存在就使用,不存在就创建,用完后,不关闭Statement对象,而是放置于Map<String, Statement>内,供下一次使用。简言之,就是重复使用Statement对象。
3)BatchExecutor:执行update(没有select,JDBC批处理不支持select),将所有sql都添加到批处理中(addBatch()),等待统一执行(executeBatch()),它缓存了多个Statement对象,每个Statement对象都是addBatch()完毕后,等待逐一执行executeBatch()批处理。与JDBC批处理相同。

9、MyBatis动态sql有什么用?
Mybatis动态sql可以在Xml映射文件内,以标签的形式编写动态sql,执行原理是根据表达式的值完成逻辑判断 并动态调整sql的功能。
Mybatis提供了9种动态sql标签:trim | where | set | foreach | if | choose | when | otherwise | bind

10、JDBC有几个步骤
1、加载驱动程序
2、获得数据库连接
3、创建一个Statement对象操作数据库,实现增删改查
4、获取结果集
5、关闭资源

11、MyBatis是否支持延迟加载?如果支持,它的实现原理是什么?
Mybatis仅支持association关联对象和collection关联集合对象的延迟加载,association指的就是一对一,collection指的就是一对多查询。在Mybatis配置文件中,可以配置是否启用延迟加载lazyLoadingEnabled=true|false。
它的原理是,使用CGLIB创建目标对象的代理对象,当调用目标方法时,进入拦截器方法,比如调用a.getB().getName(),拦截器invoke()方法发现a.getB()是null值,那么就会单独发送事先保存好的查询关联B对象的sql,把B查询上来,然后调用a.setB(b),于是a的对象b属性就有值了,接着完成a.getB().getName()方法的调用。这就是延迟加载的基本原理。
当然了,不光是Mybatis,几乎所有的包括Hibernate,支持延迟加载的原理都是一样的。

12、MyBatis的一级、二级缓存?
1)一级缓存: 基于 PerpetualCache 的 HashMap 本地缓存,其存储作用域为 Session,当 Session flush 或 close 之后,该 Session 中的所有 Cache 就将清空,默认打开一级缓存。
2)二级缓存与一级缓存其机制相同,默认也是采用 PerpetualCache,HashMap 存储,不同在于其存储作用域为 Mapper(Namespace),并且可自定义存储源,如 Ehcache。默认不打开二级缓存,要开启二级缓存,使用二级缓存属性类需要实现Serializable序列化接口(可用来保存对象的状态),可在它的映射文件中配置 ;
3)对于缓存数据更新机制,当某一个作用域(一级缓存 Session/二级缓存Namespaces)的进行了C/U/D 操作后,默认该作用域下所有 select 中的缓存将被 clear。
详解请看:https://blog.csdn.net/java123456111/article/details/125005466
13、MyBatis中如何指定使用哪一种Executor执行器?
● 在MyBatis配置文件中,在设置(settings)可以指定默认的ExecutorType执行器类型,也可以手动给DefaultSqlSessionFactory的创建SqlSession的方法传递ExecutorType类型参数。 如 SqlSession openSession(ExecutorType execType)。
● 配置默认的执行器。SIMPLE 就是普通的执行器;REUSE 执行器会重用预处理语句(prepared statements); BATCH 执行器将重用语句并执行批量更新。

14、在Mapper中如何传递多个参数
方法1:@Param注解传参法
public User selectUser(@Param(“userName”) String name, int @Param(“deptId”) deptId);

select * from user where user_name = #{userName} and dept_id = #{deptId}

● #{}里面的名称对应的是注解@Param括号里面修饰的名称。
● 这种方法在参数不多的情况还是比较直观的,(推荐使用)

方法2:Map传参法
public User selectUser(Map<String, Object> params);

select * from user where user_name = #{userName} and dept_id = #{deptId}

● #{}里面的名称对应的是Map里面的key名称
● 这种方法适合传递多个参数,且参数易变能灵活传递的情况。(推荐使用)

方法3:Java Bean传参法
public User selectUser(User user);

select * from user where user_name = #{userName} and dept_id = #{deptId}

● #{}里面的名称对应的是User类里面的成员属性 ;
● 这种方法直观,需要建一个实体类,扩展不容易,需要加属性,但代码可读性强,业务逻辑处理方 便,推荐使用。(推荐使用)。

15、如何获取生成的主键
● 新增标签中添加: keyProperty=" ID " 即可

insert into user( user_name, user_password, create_time)
values(#{userName}, #{userPassword} , #{createTime, jdbcType=TIMESTAMP})

16、Mybatis是如何将sql执行结果封装为目标对象并返回的?都有哪些映射形式?
● 第一种是使用标签,逐一定义列名和对象属性名之间的映射关系。
● 第二种是使用sql列的别名功能,将列别名书写为对象属性名,比如T_NAME AS NAME,对象属性名一般是name,小写,但是列名不区分大小写,Mybatis会忽略列名大小写,智能找到与之对应对象属性名,你甚至可以写成T_NAME AS NaMe,Mybatis一样可以正常工作。
有了列名与属性名的映射关系后,Mybatis通过反射创建对象,同时使用反射给对象的属性逐一赋值并返回,那些找不到映射关系的属性,是无法完成赋值的。

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值