关于mybatis的几个问题


一、MyBatis 中 #{}和 ${}的区别是什么?

#{}是预编译处理,$ {}表示字符串拼接。mysql的引擎会将#{}自动翻译成?然后通过preparedStatement将参数进行预编译,这样可以预防sql注入,而${}无法预防字符串注入。

二、MyBatis 有几种分页方式?它们的区别是什么?

主要有两种分页方式:

  • 逻辑分页:是通过内置的RowBounds进行分页。一次性查询很多数据,然后再在结果中检索分页的数据,这样会加大内存的消耗,容易造成内存溢出,对数据库的压力也比较大。
  • 物理分页:通过手动编码sql来查询指定条数的数据,也可以通过PageHelper分页插件来实现。这种分页方式可以有效的防止内存溢出

三、RowBounds 是一次性查询全部结果吗?为什么?

不是的,RowBounds不会一次性查出所有数据,由于mybatis是对jdbc的封装,而jdbc的驱动中有Fetch Size的配置,这个配置控制了一次性能从数据库中查询的最大数据量,一般来说这个参数的默认值是1000,也就是说每次最多从数据库中查出1000条数据,当你点击next的时候,它才会从数据库中加载另外1000条数据(如果还有一千条的话),这样就可以有效的降低内存溢出的风险。

四、MyBatis 是否支持延迟加载?延迟加载的原理是什么?

MyBatis是支持延迟加载的,设置 lazyLoadingEnabled=true 即可。
延迟加载的原理的是调用的时候触发加载,而不是在初始化的时候就加载信息。例如a.getB().getName(),a对象中并没有b对象的数据,此时调用getB()方法时,就会去调用jdbc中预存的查询sql ,会单独触发事先保存好的关联 b 对象的 SQL,从数据库中将b对象的相关数据查询出来,并用setter方法给其赋值,此时再去获取b对象的name属性就能拿到数据了。以上就是延迟加载的基本原理。

五、说一下 MyBatis 的一级缓存和二级缓存?

一级缓存:基于 PerpetualCache 的 HashMap 本地缓存,它的声明周期是和 SQLSession 一致的,有多个 SQLSession 或者分布式的环境中数据库操作,可能会出现脏数据。当 Session flush 或 close 之后,该 Session 中的所有 Cache 就将清空,默认一级缓存是开启的。
二级缓存:也是基于 PerpetualCache 的 HashMap 本地缓存,不同在于其存储作用域为 Mapper 级别的,如果多个SQLSession之间需要共享缓存,则需要使用到二级缓存,并且二级缓存可自定义存储源,如 Ehcache。默认不打开二级缓存,要开启二级缓存,使用二级缓存属性类需要实现 Serializable 序列化接口(可用来保存对象的状态)。

六、MyBatis 和 hibernate 的区别有哪些?

1、灵活性:MyBatis 更加灵活,自己可以写 SQL 语句,使用起来比较方便。
2、可移植性:MyBatis 有很多自己写的 SQL,因为每个数据库的 SQL 可以不相同,所以可移植性比较差。
3、学习和使用门槛:MyBatis 入门比较简单,使用门槛也更低。
4、二级缓存:hibernate 拥有更好的二级缓存,它的二级缓存可以自行更换为第三方的二级缓存。

七、MyBatis 有哪些执行器(Executor)?

SimpleExecutor:每执行一次 update 或 select 就开启一个 Statement 对象,用完立刻关闭 Statement 对象;
ReuseExecutor:执行 update 或 select,以 SQL 作为 key 查找 Statement 对象,存在就使用,不存在就创建,用完后不关闭 Statement 对象,而是放置于 Map 内供下一次使用。简言之,就是重复使用 Statement 对象;
BatchExecutor:执行 update(没有 select,jdbc 批处理不支持 select),将所有 SQL 都添加到批处理中(addBatch()),等待统一执行(executeBatch()),它缓存了多个 Statement 对象,每个 Statement 对象都是 addBatch()完毕后,等待逐一执行 executeBatch()批处理,与 jdbc 批处理相同。

八、MyBatis 分页插件的实现原理是什么?

分页插件的基本原理是使用 MyBatis 提供的插件接口,实现自定义插件,在插件的拦截方法内拦截待执行的 SQL,然后重写 SQL,根据 dialect 方言,添加对应的物理分页语句和物理分页参数。

九、MyBatis 如何编写一个自定义插件?

自定义插件实现原理
MyBatis 自定义插件针对 MyBatis 四大对象(Executor、StatementHandler、ParameterHandler、ResultSetHandler)进行拦截:
Executor:拦截内部执行器,它负责调用 StatementHandler 操作数据库,并把结果集通过 ResultSetHandler 进行自动映射,另外它还处理了二级缓存的操作;
StatementHandler:拦截 SQL 语法构建的处理,它是 MyBatis 直接和数据库执行 SQL 脚本的对象,另外它也实现了 MyBatis 的一级缓存;
ParameterHandler:拦截参数的处理;
ResultSetHandler:拦截结果集的处理。

扫描二维码关注公众号,回复: 12514682 查看本文章

自定义插件实现关键
MyBatis 插件要实现 Interceptor 接口,接口包含的方法,如下:

public interface Interceptor {
    
    
	Object intercept(Invocation invocation) throws Throwable;
	Object plugin(Object target);
	void setProperties(Properties properties);
}

setProperties 方法是在 MyBatis 进行配置插件的时候可以配置自定义相关属性,即:接口实现对象的参数配置;
plugin 方法是插件用于封装目标对象的,通过该方法我们可以返回目标对象本身,也可以返回一个它的代理,可以决定是否要进行拦截进而决定要返回一个什么样的目标对象,官方提供了示例:return Plugin. wrap(target, this);
intercept 方法就是要进行拦截的时候要执行的方法。
自定义插件实现示例
官方插件实现:

@Intercepts({
    
    @Signature(type = Executor. class, method = "query",
args = {
    
    MappedStatement. class, Object. class, RowBounds. class, ResultHandler. class})})
public class TestInterceptor implements Interceptor {
    
    
	public Object intercept(Invocation invocation) throws Throwable {
    
    
		Object target = invocation. getTarget(); //被代理对象
		Method method = invocation. getMethod(); //代理方法
		Object[] args = invocation. getArgs(); //方法参数
		// do something . . . . . . 方法拦截前执行代码块
		Object result = invocation. proceed();
		// do something . . . . . . . 方法拦截后执行代码块
		return result;
	}
	
	public Object plugin(Object target) {
    
    
		return Plugin. wrap(target, this);
	}
}

十、说一下MyBatis 的初始化原理、执行原理

初始化原理:

  1. 调用SqlSessionFactoryBuilder对象的build(inputStream)方法;
  2. SqlSessionFactoryBuilder会依据输入流inputStream等信息创建XMLConfigBuilder对象;
  3. SqlSessionFactoryBuilder调用XMLConfigBuilder对象的parse()方法;
  4. XMLConfigBuilder对象返回Configuration对象;
  5. SqlSessionFactoryBuilder依据Configuration对象创建一个DefaultSessionFactory对象;
  6. SqlSessionFactoryBuilder返回 DefaultSessionFactory对象给Client,供Client使用;

执行原理:
第1步,读取配置文件缓存到Configuration对象,用于创建SqlSessionFactory;
第2步,SqlSession的执行过程,主要是是运用了反射机制和动态代理来完成jdbc的底层调用。

SqlSessionFactoryBuilder : SqlSessionFactory的构造器,用于创建SqlSessionFactory。採用了Builder设计模式
Configuration :该对象是mybatis-config.xml文件里全部mybatis配置信息
SqlSessionFactory:SqlSession工厂类。以工厂形式创建SqlSession对象,採用了Factory工厂设计模式
XmlConfigParser :负责将mybatis-config.xml配置文件解析成Configuration对象,共SqlSessonFactoryBuilder使用,创建SqlSessionFactory

猜你喜欢

转载自blog.csdn.net/qq_42697271/article/details/113853463