Java 面试题-Hibernate、MyBatis

本文探讨了Hibernate的封装优势、ORM概念、SQL查看方法,以及它与MyBatis的差异,包括查询方式、缓存机制、状态管理、执行效率和分页策略。同时,深入解析了MyBatis的逻辑分页、物理分页和缓存机制。最后,比较了两者在设计模式和适用场景上的特点。
摘要由CSDN通过智能技术生成
1.为什么要使用 Hibernate?
  • Hibernate是对 jdbc 的封装,大大简化了数据访问层的繁琐的重复性代码。
  • Hibernate是一个优秀的 ORM 实现,很多程度上简化了 DAO 层的编码功能。
  • 可以很方便的进行数据库的移植工作。
  • 提供了缓存机制,是程序执行更改的高效。
2.什么是 ORM 框架?

    ORM(Object Relation Mapping)对象关系映射,是把数据库中的关系数据映射成为程序中的对象。

    使用 ORM 的优点:提高了开发效率降低了开发成本、开发更简单更对象化、可移植更强。

3.Hibernate 中如何在控制台查看打印的 SQL 语句?

    在 Config 里面把 hibernate. show_SQL 设置为 true 就可以。但不建议开启,开启之后会降低程序的运行效率。

4.Hibernate 有几种查询方式?

    三种:hql、原生 SQL、条件查询 Criteria。

5.Hibernate 实体类可以被定义为 final 吗?

    实体类可以定义为 final 类,但这样的话就不能使用 hibernate 代理模式下的延迟关联提供性能了,所以不建议定义实体类为 final。

6.在 Hibernate 中使用 Integer 和 int 做映射有什么区别?

    Integer 类型为对象,它的值允许为 null,而 int 属于基础数据类型,值不能为 null。

7.Hibernate 是如何工作的?
  • 读取并解析配置文件。
  • 读取并解析映射文件,创建 SessionFactory。
  • 打开 Session。
  • 创建事务。
  • 进行持久化操作。
  • 提交事务。
  • 关闭 Session。
  • 关闭 SessionFactory。
8.get() 和 load() 的区别?
  • 数据查询时,没有 OID 指定的对象,get() 返回 null;load() 返回一个代理对象。
  • load()支持延迟加载;get() 不支持延迟加载。
9.说一下 Hibernate 的缓存机制?

    Hibernate 常用的缓存有一级缓存和二级缓存:

  • 一级缓存:也叫 Session 缓存,只在 Session 作用范围内有效,不需要用户干涉,由 hibernate 自身维护,可以通过:evict(object)清除 object 的缓存;clear()清除一级缓存中的所有缓存;flush()刷出缓存;

  • 二级缓存:应用级别的缓存,在所有 Session 中都有效,支持配置第三方的缓存,如:EhCache。

10.Hibernate 对象有哪些状态?
  • 临时/瞬时状态:直接 new 出来的对象,该对象还没被持久化(没保存在数据库中),不受 Session 管理。
  • 持久化状态:当调用 Session 的 save/saveOrupdate/get/load/list 等方法的时候,对象就是持久化状态。
  • 游离状态:Session 关闭之后对象就是游离状态。
11.在 Hibernate 中 getCurrentSession 和 openSession 的区别是什么?
  • getCurrentSession 会绑定当前线程,而 openSession 则不会。
  • getCurrentSession 事务是 Spring 控制的,并且不需要手动关闭,而 openSession 需要我们自己手动开启和提交事务。
12.Hibernate 实体类必须要有无参构造函数吗?为什么?

    Hibernate 中每个实体类必须提供一个无参构造函数,因为 hibernate 框架要使用 reflection api,通过调用 ClassnewInstance() 来创建实体类的实例,如果没有无参的构造函数就会抛出异常。

13.MyBatis 中 #{}和 ${}的区别是什么?

    #{}是预编译处理,${}是字符替换。 在使用 #{}时,MyBatis 会将 SQL 中的 #{}替换成“?”,配合 PreparedStatement 的 set 方法赋值,这样可以有效的防止 SQL 注入,保证程序的运行安全。

14.MyBatis 有几种分页方式?

    分页方式:逻辑分页和物理分页。

  • 逻辑分页: 使用 MyBatis 自带的 RowBounds 进行分页,它是一次性查询很多数据,然后在数据中再进行检索。
  • 物理分页: 自己手写 SQL 分页或使用分页插件 PageHelper,去数据库查询指定条数的分页数据的形式。
15.RowBounds 是一次性查询全部结果吗?为什么?

    RowBounds 表面是在“所有”数据中检索数据,其实并非是一次性查询出所有数据,因为 MyBatis 是对 jdbc 的封装,在 jdbc 驱动中有一个 Fetch Size 的配置,它规定了每次最多从数据库查询多少条数据,假如你要查询更多数据,它会在你执行 next()的时候,去查询更多的数据。就好比你去自动取款机取 10000 元,但取款机每次最多能取 2500 元,所以你要取 4 次才能把钱取完。只是对于 jdbc 来说,当你调用 next()的时候会自动帮你完成查询工作。这样做的好处可以有效的防止内存溢出。

16.MyBatis 逻辑分页和物理分页的区别是什么?
  • 逻辑分页是一次性查询很多数据,然后再在结果中检索分页的数据。这样做弊端是需要消耗大量的内存、有内存溢出的风险、对数据库压力较大。
  • 物理分页是从数据库查询指定条数的数据,弥补了一次性全部查出的所有数据的种种缺点,比如需要大量的内存,对数据库查询压力较大等问题。
17.MyBatis 是否支持延迟加载?延迟加载的原理是什么?

    MyBatis 支持延迟加载,设置 lazyLoadingEnabled=true 即可。

    延迟加载的原理的是调用的时候触发加载,而不是在初始化的时候就加载信息。比如调用 a. getB(). getName(),这个时候发现 a. getB() 的值为 null,此时会单独触发事先保存好的关联 B 对象的 SQL,先查询出来 B,然后再调用 a. setB(b),而这时候再调用 a. getB(). getName() 就有值了,这就是延迟加载的基本原理。

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

    开启二级缓存数据查询流程:二级缓存 -> 一级缓存 -> 数据库。

    缓存更新机制:当某一个作用域(一级缓存 Session/二级缓存 Mapper)进行了C/U/D 操作后,默认该作用域下所有 select 中的缓存将被 clear。

19.MyBatis 和 hibernate 的区别有哪些?
  • 灵活性:MyBatis 更加灵活,自己可以写 SQL 语句,使用起来比较方便。
  • 可移植性:MyBatis 有很多自己写的 SQL,因为每个数据库的 SQL 可以不相同,所以可移植性比较差。
  • 学习和使用门槛:MyBatis 入门比较简单,使用门槛也更低。
  • 二级缓存:hibernate 拥有更好的二级缓存,它的二级缓存可以自行更换为第三方的二级缓存。
20.MyBatis 有哪些执行器(Executor)?

    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 批处理相同。
21.MyBatis 分页插件的实现原理是什么?

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

22.MyBatis 如何编写一个自定义插件?
  1. 自定义插件实现原理
    MyBatis 自定义插件针对 MyBatis 四大对象(Executor、StatementHandler、ParameterHandler、ResultSetHandler)进行拦截:

    • Executor:拦截内部执行器,它负责调用 StatementHandler 操作数据库,并把结果集通过 ResultSetHandler 进行自动映射,另外它还处理了二级缓存的操作;
    • StatementHandler:拦截 SQL 语法构建的处理,它是 MyBatis 直接和数据库执行 SQL 脚本的对象,另外它也实现了 MyBatis 的一级缓存;
    • ParameterHandler:拦截参数的处理;
    • ResultSetHandler:拦截结果集的处理。
  2. 自定义插件实现关键
    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 方法就是要进行拦截的时候要执行的方法。
  3. 自定义插件实现示例
    官方插件实现:

    @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);
        }
    }
    
23.MyBatis 使用了哪些设计模式?
  1. 建造者模式:SqlSessionFactoryBuilder、XMLConfigBuilder、XMLMapperBuilder、XMLStatementBuilder、CacheBuilder;
  2. 工厂模式:SqlSessionFactory、ObjectFactory、MapperProxyFactory;
  3. 单例模式:ErrorContext 和 LogFactory;
  4. 代理模式:Mybatis实现的核心,比如 MapperProxy、ConnectionLogger,用的 jdk 的动态代理;还有 executor.loader 包使用了 cglib 或者 javassist 达到延迟加载的效果;
  5. 组合模式:例如 SqlNode 和 各个子类 ChooseSqlNode 等;
  6. 模板方法模式:例如 BaseExecutor 和 SimpleExecutor,还有 BaseTypeHandler 和所有的子类例如 IntegerTypeHandler;
  7. 适配器模式:例如 Log 的 Mybatis 接口和它对 jdbc、log4j 等各种日志框架的适配实现;
  8. 装饰者模式:Cache 的实现类 LruCache、FifoCache 等都是装饰一个类PerpetualCache;
  9. 迭代器模式:PropertyTokenizer;
24.MyBatis 除了 select、insert、update、delete外还有哪些常用标签?
  • sqlMap:Mybatis 配置文件的顶级标签;
  • resultMap:定义数据库表与实体类的映射关系;
  • result:resultMap 的子标签;
  • sql:定义代码片段;
  • include:引入 sql 标签定义的代码片段;
  • if:条件判断
  • foreach:foreach标签主要用于构建in条件,可在sql中对集合进行迭代;
  • choose:按顺序判断when中的条件出否成立;
  • where:动态配置 where 关键字;
  • set:动态配置 set 关键字;
  • trim:格式化输出,也可以通过trim标签设定或忽略前后缀来实现;
  • collection、association:配置多张表的关联关系;
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值