面试必会 --> MyBatis篇

mybatis · 浏览次数 : 0

小编点评

本文主要介绍了 MyBatis 框架的特点、优点、缺点以及实现细节。 **一、MyBatis框架** **1. 半ORM框架** MyBatis 是一个半ORM(对象关系映射)框架,简化了数据访问层,使开发者专注于 SQL 语句编写,而无需处理底层数据库连接和驱动等问题。 **2. 封装** MyBatis 封装了 JDBCTemplate,开发者只需编写原始 SQL 语句,而无需直接编写底层的驱动和数据库连接管理代码。 **2. 配置与映射** MyBatis 使用 XML 或注解来配置和映射原生信息,使得 POJO 可以映射成数据库中的记录。 **3. 性能优势** 由于 MyBatis 可控性强,开发者可以灵活地编写 SQL 语句并严格控制执行性能。 **二、MyBatis优点与缺点** **1. 优点:** * SQL 编写更简单 * 减少了重复的代码量 * 良好的数据库兼容性 * 开发效率较高 * 支持与 Spring 集成 * 支持一级和二级缓存 **2. 缺点:** * SQL 编写可能较繁琐,尤其是在包含大量字段和复杂关联时 * 数据库移植性相对较差,因为 SQL 语句依赖于数据库类型 **三、特殊概念解释** * #{}:预编译处理,避免SQL注入 * ${}:字符串替换,适用于不需要预编译的场景 * RowBounds:用于实现物理分页 **四、MyBatis实现细节** * 延迟加载:只有 association 和 collection 关联对象支持 * 分布式缓存:可以使用 Redis、Memcached 等分布式缓存 **五、其他数据库操作方法** * 动态 SQL:通过标签引入 sql 片段来实现 * MyBatis缓存实现:包括一级缓存(LocalCache)和二级缓存(CachingExecutor) **六、JDBC 编程步骤** 1. 加载 JDBC 驱动并进行初始化。 2. 建立 JDBC 和数据库之间的 Connection 连接。 3. 创建 Statement 或 PreparedStatement 接口,执行 SQL 语句。 4. 处理和显示结果。 5. 释放资源。

正文

什么是MyBatis

  1. Mybatis是一个半ORM(对象关系映射)框架,它内部封装了JDBC,开发时只需要关注SQL语句本身,不需要花费精力去处理加载驱动、创建连接、创建statement等繁杂的过程。程序员直接编写原生态sql,可以严格控制sql执行性能,灵活度高。

  2. MyBatis 可以使用 XML 或注解来配置和映射原生信息,将 POJO映射成数据库中的记录,避免了几乎所有的 JDBC 代码和手动设置参数以及获取结果集。

  3. 通过xml 文件或注解的方式将要执行的各种 statement 配置起来,并通过java对象和statement中sql的动态参数进行映射生成最终执行的sql语句,最后由mybatis框架执行sql并将结果映射为java对象并返回。(从执行sql到返回result的过程)。

说说MyBatis的优点和缺点

优点:

  • 基于SQL语句编程,相当灵活,不会对应用程序或者数据库的现有设计造成任何影响,SQL写在XML里,解除sql与程序代码的耦合,便于统一管理;提供XML标签,支持编写动态SQL语句,并可重用。
  • 与JDBC相比,减少了50%以上的代码量,消除了JDBC大量冗余的代码,不需要手动开关连接;
  • 很好的与各种数据库兼容(因为MyBatis使用JDBC来连接数据库,所以只要JDBC支持的数据库MyBatis都支持)。
  • 能够与Spring很好的集成;
  • 提供映射标签,支持对象与数据库的ORM字段关系映射;提供对象关系映射标签,支持对象关系组件维护。

缺点:

  • SQL语句的编写工作量较大,尤其当字段多、关联表多时,对开发人员编写SQL语句的功底有一定要求。
  • SQL语句依赖于数据库,导致数据库移植性差,不能随意更换数据库。

#{}和${}的区别是什么?

  • #{}是预编译处理,${}是字符串替换。
  • Mybatis在处理#{}时,会将sql中的#{}替换为?号,调用PreparedStatement的set方法来赋值;
  • Mybatis在处理${}时,就是把\${}替换成变量的值。
  • 使用#{}可以有效的防止SQL注入,提高系统安全性。

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

  • 第1种: 通过在查询的sql语句中定义字段名的别名,让字段名的别名和实体类的属性名一致.

      <select id=”selectorder” parametertype=”int” resultetype=”me.gacl.domain.order”>
       select order_id id, order_no orderno ,order_price price form orders where
      order_id=#{id}; 
       </select>
    

  • 第2种: 通过来映射字段名和实体类属性名的一一对应的关系。

       <select id="getOrder" parameterType="int" resultMap="orderresultmap">
       select * from orders where order_id=#{id} 
       </select> 
    
       <resultMap type=”me.gacl.domain.order” id=”orderresultmap”> 
       <!–用id属性来映射主键字段–> 
       <id property=”id” column=”order_id”> 
    
       <!–用result属性来映射非主键字段,property为实体类属性名,column为数据表中的属性–> 
       <result property = “orderno” column =”order_no”/> 
       <result property=”price” column=”order_price” /> 
       </reslutMap>
    

Mybatis是如何进行分页的?分页插件的原理是什么?

Mybatis使用RowBounds对象进行分页,它是针对ResultSet结果集执行的内存分页,而非物理分页。可以在sql内直接拼写带有物理分页的参数来完成物理分页功能,也可以使用分页插件来完成物理分页,比如:MySQL数据的时候,在原有SQL后面拼写limit。

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


Mybatis是如何将sql执行结果封装为目标对象并返回的?都有哪些映射形式?

  • 第一种是使用标签,逐一定义数据库列名和对象属性名之间的映射关系。
  • 第二种是使用sql列的别名功能,将列的别名书写为对象属性名。

有了列名与属性名的映射关系后,Mybatis通过反射创建对象,同时使用反射给对象的属性逐一赋值并返回,那些找不到映射关系的属性,是无法完成赋值的。


如何执行批量插入?

首先,创建一个简单的insert语句:

	<insert id=”insertname”> 
	 insert into names (name) values (#{value}) 
	 </insert>

然后在java代码中像下面这样执行批处理插入:

	 list<string> names = new arraylist(); 
	 names.add(“fred”); 
	 names.add(“barney”); 
	 names.add(“betty”); 
	 names.add(“wilma”); 

	 // 注意这里 executortype.batch 
	 sqlsession sqlsession = sqlsessionfactory.opensession(executortype.batch);
	 try { 
	 namemapper mapper = sqlsession.getmapper(namemapper.class);
	 for (string name : names) { 
	 mapper.insertname(name); 
	 } 
	 sqlsession.commit(); 
	 }catch(Exception e){ 
	 e.printStackTrace(); 
	 sqlSession.rollback(); 
	 throw e; 
	 } 
	 finally { 
	 sqlsession.close(); 
	 }

Xml映射文件中,除了常见的select|insert|updae|delete标签之外,还有哪些标签?

加上动态sql的9个标签,其中为sql片段标签,通过标签引入sql片段,为不支持自增的主键生成策略标签。


MyBatis实现一对一有几种方式?具体怎么操作的?

有联合查询和嵌套查询,联合查询是几个表联合查询,只查询一次, 通过在resultMap里面配置association节点配置一对一的类就可以完成;

嵌套查询是先查一个表,根据这个表里面的结果的 外键id,去再另外一个表里面查询数据,也是通过association配置,但另外一个表的查询通过select属性配置。


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,支持延迟加载的原理都是一样的。


说说Mybatis的缓存机制:

Mybatis整体:


image


一级缓存localCache
在应用运行过程中,我们有可能在一次数据库会话中,执行多次查询条件完全相同的 SQL,MyBatis 提供了一级缓存的方案优化这部分场景,如果是相同的 SQL 语句,会优先命中一级缓存,避免直接对数据库进行查询,提高性能。

每个 SqlSession 中持有了 Executor,每个 Executor 中有一个 LocalCache。当用户发起查询时,MyBatis 根据当前执行的语句生成 MappedStatement,在 Local Cache 进行查询,如果缓存命中的话,直接返回结果给用户,如果缓存没有命中的话,查询数据库,结果写入 Local Cache,最后返回结果给用户。具体实现类的类关系图如下图所示:


image


  • MyBatis 一级缓存的生命周期和 SqlSession 一致。
  • MyBatis 一级缓存内部设计简单,只是一个没有容量限定的 HashMap,在缓存的功能性上有所欠缺。
  • MyBatis 的一级缓存最大范围是 SqlSession 内部,有多个 SqlSession 或者分布式的环境下,数据库写操作会引起脏数据,建议设定缓存级别为 Statement。

二级缓存
在上文中提到的一级缓存中,其最大的共享范围就是一个 SqlSession 内部,如果多个 SqlSession之间需要共享缓存,则需要使用到二级缓存。开启二级缓存后,会使用 CachingExecutor 装饰Executor,进入一级缓存的查询流程前,先在 CachingExecutor 进行二级缓存的查询,具体的工作流程如下所示。


image


二级缓存开启后,同一个 namespace 下的所有操作语句,都影响着同一个 Cache,即二级缓存被多个 SqlSession 共享,是一个全局的变量。
当开启缓存后,数据的查询执行的流程为:

二级缓存 -> 一级缓存 -> 数据库

  • MyBatis 的二级缓存相对于一级缓存来说,实现了 SqlSession 之间缓存数据的共享,同时粒度更加细,能够到 namespace 级别,通过 Cache 接口实现类不同的组合,对 Cache 的可控性也更强。
  • MyBatis 在多表查询时,极大可能会出现脏数据,有设计上的缺陷,安全使用二级缓存的条件比较苛刻。
  • 在分布式环境下,由于默认的 MyBatis Cache 实现都是基于本地的,分布式环境下必然会出现读取到脏数据,需要使用集中式缓存将 MyBatis 的 Cache 接口实现,有一定的开发成本,直接使用 Redis、Memcached 等分布式缓存可能成本更低,安全性也更高。

JDBC 编程有哪些步骤?

  1. 装载相应的数据库的 JDBC 驱动并进行初始化:

     Class.forName("com.mysql.jdbc.Driver");
    

  1. 建立 JDBC 和数据库之间的 Connection 连接:

     Connection c = DriverManager.getConnection("jdbc:mysql://127.0.0.1:3306/test?
     characterEncoding=UTF-8", "root", "123456");
    

  1. 创建 Statement 或者 PreparedStatement 接口,执行 SQL 语句。
  2. 处理和显示结果。
  3. 释放资源。

MyBatis 中见过什么设计模式?


image


MyBatis 中比如 UserMapper.java 是接口,为什么没有实现类还能调用?

使用JDK动态代理+MapperProxy。本质上调用的是MapperProxy的invoke方法。

与面试必会 --> MyBatis篇相似的内容:

面试必会 --> MyBatis篇

什么是MyBatis Mybatis是一个半ORM(对象关系映射)框架,它内部封装了JDBC,开发时只需要关注SQL语句本身,不需要花费精力去处理加载驱动、创建连接、创建statement等繁杂的过程。程序员直接编写原生态sql,可以严格控制sql执行性能,灵活度高。 MyBatis 可以使用 XM

大数据面试SQL每日一题系列:最高峰同时在线主播人数。字节,快手等大厂高频面试题

大数据面试SQL每日一题系列:最高峰同时在线主播人数。字节,快手等大厂高频面试题 之后会不定期更新每日一题sql系列。 SQL面试题每日一题系列内容均来自于网络以及实际使用情况收集,如有雷同,纯属巧合。 1.题目 问题1:如下为某直播平台各主播的开播及关播时间数据明细,现在需要计算该平台最高峰期同时

Java并发篇:6个必备的Java并发面试种子题目

免费体验AI绘画:https://www.topgpt.one;文章涉及了几个常见的并发编程相关的主题。首先,线程的创建和生命周期是面试中常被问及的话题,面试官可能会询问如何创建线程、线程的状态转换以及如何控制线程的执行顺序等。其次,synchronized关键字是用于实现线程同步的重要工具,面试中可能会涉及到它的使用场景以及与其他同步机制的比较。此外,抽象队列同步器(AQS)是Java并发编程中

万字长文详解Java线程池面试题

大家好,我是王有志。今天是《面霸的自我修养》第 6 篇文章,我们一起来看看面试中会问到哪些关于线程池的问题吧。

[转帖]面渣逆袭:二十二图、八千字、二十问,彻底搞定MyBatis!

https://cdn.modb.pro/db/334793 大家好,我是老三,面渣逆袭系列继续,这节我们的主角是MyBatis,作为当前国内最流行的ORM框架,是我们这些crud选手最趁手的工具,赶紧来看看面试都会问哪些问题吧。 基础 1.说说什么是MyBatis? MyBatis logo 先吹

携程Java三面面经,已拿 offer!!

分享一位读者投稿的携程校招 Java 岗位的面经。 下面是正文。 个人背景:双非本,机械专业转码。 携程在正式面试之前,会有一个性格测试(40分钟)。性格测试之后,大概过一周进行笔试。笔试之后,会邮件通知是否通过并预约第一轮面试时间。 普通 offer 一般只有两面,如果是 sp 或者 ssp 的话

空闲空间管理和文件系统结构的优化策略

对于有科班背景的读者,可以跳过本系列文章。这些文章的主要目的是通过简单易懂的汇总,帮助非科班出身的读者理解底层知识,进一步了解为什么在面试中会涉及这些底层问题。否则,某些概念将始终无法理解。这些计算机基础文章将为你打通知识的任督二脉,祝你在编程领域中取得成功!

探索计算机的I/O控制方式:了解DMA控制器的作用与优势

对于有科班背景的读者,可以跳过本系列文章。这些文章的主要目的是通过简单易懂的汇总,帮助非科班出身的读者理解底层知识,进一步了解为什么在面试中会涉及这些底层问题。否则,某些概念将始终无法理解。这些计算机基础文章将为你打通知识的任督二脉,祝你在编程领域中取得成功!

阿里面试:NIO为什么会导致CPU100%?

在 Java 中总共有三种 IO 类型:BIO(Blocking I/O,阻塞I/O)、NIO(Non-blocking I/O,非阻塞I/O)和 AIO(Asynchronous I/O,异步I/O),它们的区别如下: 在 JDK 1.4 之前,只有 BIO 一种模式,其开发过程相对简单,新来一个

[转帖]MySQL优化的5个维度

面试官如果问你:你会从哪些维度进行MySQL性能优化?你会怎么回答?所谓的性能优化,一般针对的是MySQL查询的优化。既然是优化查询,我们自然要先知道查询操作要经过哪些环节,然后思考可以在哪些环节进行优化。我之前写过一条SQL查询语句是如何执行的?,感兴趣的朋友可以阅读一下,我用其中的一张图展示查询