03---java面试八股文——mybatis-------8题

本文详细介绍了MyBatis中一对一和一对多查询的实现方式,包括嵌套Select、嵌套结果映射、延迟加载,以及一级和二级缓存的使用。同时讲解了接口绑定的不同实现和MyBatis插件的运行原理。
摘要由CSDN通过智能技术生成

21、MyBatis实现一对一查询

  1. MyBatis 有两种不同的方式加载关联:

    • 嵌套 Select 查询:通过执行另外一个 SQL 映射语句来加载期望的复杂类型。
    • 嵌套结果映射:使用嵌套的结果映射来处理连接结果的重复子集。
    • 查看mybatis的关联
      MyBatis是一种流行的Java持久化框架,它提供了多种方式来实现一对一关联。以下是几种常见的方式:
  2. 嵌套select查询(Nested Queries):这种方式通过在主查询中使用子查询来获取关联对象的数据。在MyBatis中,可以使用<select>标签嵌套另一个<select>标签来实现一对一关联查询。具体操作是在主查询中使用子查询获取关联对象的数据,并将其映射到主对象的属性中。

<resultMap id="blogResult" type="Blog">
  <association property="author" column="author_id" javaType="Author" select="selectAuthor"/>
</resultMap>

<select id="selectBlog" resultMap="blogResult">
  SELECT * FROM BLOG WHERE ID = #{id}
</select>

<select id="selectAuthor" resultType="Author">
  SELECT * FROM AUTHOR WHERE ID = #{id}
</select>
  1. 嵌套结果映射(Nested Result Maps):这种方式通过在结果映射中定义嵌套的结果映射来实现一对一关联。在MyBatis中,可以使用<resultMap>标签定义嵌套的结果映射,然后在主结果映射中引用它。具体操作是在主结果映射中使用<association>标签引用关联对象的结果映射。

    • 延迟加载(Lazy Loading):这种方式通过延迟加载关联对象的数据来实现一对一关联。在MyBatis中,可以使用<association>标签的fetchType属性设置为lazy来启用延迟加载。具体操作是在需要访问关联对象数据时才进行加载。
  2. 嵌套查询和嵌套结果映射的组合:有时候,可以将嵌套查询和嵌套结果映射结合起来使用,以实现更复杂的一对一关联查询。

在这里插入图片描述

22、MyBatis实现一对多查询有几种方式,怎么操作的?((关联(association))

MyBatis是一种Java持久层框架,它提供了多种方式来实现一对多的关系。以下是几种常见的方式及其操作方法:

  1. 嵌套select查询(Nested Queries):通过在主查询中嵌套子查询来获取关联对象的数据。在MyBatis的Mapper XML文件中,可以使用<collection>标签定义一个嵌套查询,通过指定子查询的SQL语句和结果映射关系来实现一对多的关系。

    示例代码:

<select id="selectPostsForBlog" resultType="Post">
  SELECT * FROM POST WHERE BLOG_ID = #{id}
</select>

<resultMap id="blogResult" type="Blog">
  <collection property="posts" javaType="ArrayList" column="id" ofType="Post" select="selectPostsForBlog"/>
</resultMap>

<select id="selectBlog" resultMap="blogResult">
  SELECT * FROM BLOG WHERE ID = #{id}
</select>


1.1 . 延迟加载(Lazy Loading):在查询主对象时,不立即加载关联对象的数据,而是在需要使用关联对象时再进行加载。在MyBatis的Mapper XML文件中,可以使用<collection>标签的fetchType属性设置为lazy来实现延迟加载。

示例代码:

 <select id="selectPostsForBlog" resultType="Post">
  SELECT * FROM POST WHERE BLOG_ID = #{id}
</select>

<resultMap id="blogResult" type="Blog">
   <!-- collection标签的含义: posts 是一个存储 Post 的 ArrayList 集合-->
  <collection property="posts" javaType="ArrayList" column="id" fetchType="lazy" ofType="Post" select="selectPostsForBlog"/>
</resultMap>

<select id="selectBlog" resultMap="blogResult">
  SELECT * FROM BLOG WHERE ID = #{id}
</select>
  1. 嵌套结果映射(Nested Result Maps):通过在结果映射中定义嵌套的结果映射来实现一对多的关系。在MyBatis的Mapper XML文件中,可以使用<resultMap>标签中使用<collection>标签定义一个嵌套的结果映射。

    示例代码:

<select id="selectBlog" resultMap="blogResult">
  select
  B.id as blog_id,
  B.title as blog_title,
  B.author_id as blog_author_id,
  P.id as post_id,
  P.subject as post_subject,
  P.body as post_body,
  from Blog B
  left outer join Post P on B.id = P.blog_id
  where B.id = #{id}
</select>

<resultMap id="blogResult" type="Blog">
  <id property="id" column="blog_id" />
  <result property="title" column="blog_title"/>
  <collection property="posts" ofType="Post">
    <id property="id" column="post_id"/>
    <result property="subject" column="post_subject"/>
    <result property="body" column="post_body"/>
  </collection>
</resultMap>

这段代码和上面一段一样的作用

<select id="selectBlog" resultMap="blogResult">
  select
  B.id as blog_id,
  B.title as blog_title,
  B.author_id as blog_author_id,
  P.id as post_id,
  P.subject as post_subject,
  P.body as post_body,
  from Blog B
  left outer join Post P on B.id = P.blog_id
  where B.id = #{id}
</select>

<resultMap id="blogResult" type="Blog">
  <id property="id" column="blog_id" />
  <result property="title" column="blog_title"/>
  <collection property="posts" ofType="Post" resultMap="blogPostResult" columnPrefix="post_"/>
</resultMap>

<resultMap id="blogPostResult" type="Post">
  <id property="id" column="id"/>
  <result property="subject" column="subject"/>
  <result property="body" column="body"/>
</resultMap>

示例:一个用户上面有多个账户

public class User implements Serializable {
    private Integer id;
    private String username;
    private String address;
    private String sex;
    private Date birthday;

    //一对多关系映射:主表实体应该包含从表实体的集合引用
    private List<Account> accounts;
}


public class Account implements Serializable{
 	private Integer aid;
    private Integer uid;
    private Double money;
}
<?xml version="1.0" encoding="UTF-8"?>
<!DOCTYPE mapper
        PUBLIC "-//mybatis.org//DTD Mapper 3.0//EN"
        "http://mybatis.org/dtd/mybatis-3-mapper.dtd">

<mapper namespace="com.itheima.dao.IUserDao">

    <!--定义 user 的 resultMap -->
    <resultMap id="userAccountMap" type="user">
    
        <id property="id" column="id"></id>
        <result property="username" column="username"></result>
        <result property="address" column="address"></result>
        <result property="sex" column="sex"></result>
        
        <!--配置user对象account集合的映射
            ofType集合中元素的类型-->
        <collection property="accounts" ofType="account" fetchType="lazy">
            <id column="aid" property="id" ></id>
            <result column="uid" property="uid"></result>
            <result column="money" property="money"></result>
        </collection>
        
    </resultMap>

    <!--配置查询所有-->
    <select id="findAll" resultMap="userAccountMap">
        select u.*, a.id as aid, a.uid, a.money from user u left outer join account a on u.id = a.uid
    </select>
    
</mapper>

23、Mybatis是否支持延迟加载?如果支持,它的实现原理是什么?

  1. 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()方法的调用。这就是延迟加载的基本原理。

补充什么是 CGLIB?

  1. 什么是 CGLIB?
    • CGLIB是一个功能强大,高性能的代码生成包。它为没有实现接口的类提供代理,为JDK的动态代理提供了很好的补充。
    • 通常可以使用Java的动态代理创建代理,但当要代理的类没有实现接口或者为了更好的性能,CGLIB 是一个好的选择。
  2. CGLIB 的原理
    • 动态生成一个要代理类的子类,子类重写要代理的类的所有不是 final 的方法。在子类中采用方法拦截的技术拦截所有父类方法的调用,顺势织入横切逻辑。
  3. CGLIB 底层:采用ASM字节码生成框架,使用字节码技术生成代理类,比使用 Java 反射效率要高。

24、Mybatis的一级、二级缓存:

  1. 一级缓存: 基于 PerpetualCache 的 HashMap 本地缓存,其存储作用域为Session,当 Session flush 或 close 之后,该 Session 中的所有 Cache 就将清空,默认打开一级缓存。
  2. 二级缓存与一级缓存其机制相同,默认也是采用 PerpetualCache,HashMap存储,不同在于其存储作用域为 Mapper(Namespace),并且可自定义存储源,如 Ehcache。默认不打开二级缓存,要开启二级缓存,使用二级缓存属性类需要实现Serializable 序列化接口(可用来保存对象的状态),可在它的映射文件中配置<cache/>
  3. 对于缓存数据更新机制,当某一个作用域(一级缓存 Session/二级缓存Namespaces)的进行了 C/U/D 操作后,默认该作用域下所有 select 中的缓存将被 clear。

25、什么是MyBatis的接口绑定?有哪些实现方式?

  1. 接口绑定,就是在MyBatis 中任意定义接口,然后把接口里面的方法和 SQL语句绑定, 我们直接调用接口方法就可以,这样比起原来了SqlSession提供的方法我们可以有更加灵活的选择和设置。
  2. 接口绑定有两种实现方式:
    • 一种是通过注解绑定,就是在接口的方法上面加上@Select、@Update 等注解,里面包含 Sql 语句来绑定;
    • 另外一种就是通过 xml里面写SQL来绑定, 在这种情况下,要指定xml映射文件里面的namespace必须为接口的全路径名。当Sql语句比较简单时候,用注解绑定, 当SQL语句比较复杂时候,用 xml 绑定,一般用 xml 绑定的比较多。

26、Mapper编写有哪几种方式

  1. 第一种:接口实现类继承SqlSessionDaoSupport:使用此种方法需要编写mapper 接口,mapper 接口实现类、mapper.xml 文件。

1、在 MybatisConfig.xml 中配置 mapper.xml 的位置

<mappers>
 <mapper resource="mapper.xml 文件的地址" />
 <mapper resource="mapper.xml 文件的地址" />
 </mappers>

2、定义 mapper 接口
3、实现类继承SqlSessionDaoSupport ,mapper 方法中可以 this.getSqlSession()进行数据增删改查。
4、spring 配置

<bean id=" " class="mapper接口的实现类">
 <property name="sqlSessionFactory" ref="sqlSessionFactory"></property>
 </bean>
  1. 第二种:使用org.mybatis.spring.mapper.MapperFactoryBean:

1、在 MybatisConfig.xml 中配置 mapper.xml 的位置,如果 mapper.xml 和mappre 接口的名称相同且在同一个目录,这里可以不用配置

<mappers>
 <mapper resource="mapper.xml 文件的地址" />
 <mapper resource="mapper.xml 文件的地址" />
 </mappers>

2、定义 mapper 接口:
2.1、mapper.xml 中的 namespace 为 mapper 接口的地址
2.2、mapper 接口中的方法名和 mapper.xml 中的定义的 statement 的 id 保持一

3、Spring 中定义

<bean id="" class="org.mybatis.spring.mapper.MapperFactoryBean">
 <property name="mapperInterface" value="mapper接口地址" />
 <property name="sqlSessionFactory" ref="sqlSessionFactory" />
 </bean>
  1. 第三种:使用mapper 扫描器:

1、mapper.xml 文件编写:
mapper.xml 中的 namespace 为 mapper 接口的地址;
mapper 接口中的方法名和 mapper.xml 中的定义的 statement 的 id 保持一致;
如果将mapper.xml和mapper接口的名称保持一致则不用在Mybatis.xml中进行配置。

2、定义 mapper 接口:
注意mapper.xml 的文件名和 mapper 的接口名称保持一致,且放在同一个目录

3、配置 mapper 扫描器:

<bean class="org.mybatis.spring.mapper.MapperScannerConfigurer">
 <property name="basePackage" value="mapper 接口包地址"></property>
 <property name="sqlSessionFactoryBeanName" value="sqlSessionFactory"/>
 </bean>

4、使用扫描器后从spring 容器中获取 mapper的实现对象

27、使用MyBatis的mapper接口调用时有哪些要求?

  1. Mapper 接口方法名和 mapper.xml 中定义的每个 sql 的 id 相同;
  2. Mapper 接口方法的输入参数类型和 mapper.xml 中定义的每个 sql 的parameterType 的类型相同;
  3. Mapper 接口方法的输出参数类型和 mapper.xml 中定义的每个 sql的resultType 的类型相同;
  4. Mapper.xml 文件中的 namespace 即是 mapper 接口的类路径

28、简述Mybatis的插件运行原理,以及如何编写一个插件。

  1. Mybatis 仅可以编写针对 ParameterHandler、ResultSetHandler、StatementHandler、Executor 这 4 种接口的插件,Mybatis 使用 JDK 的动态代理,为需要拦截的接口生成代理对象以实现接口方法拦截功能,每当执行这4种接口对象的方法时,就会进入拦截方法,具体就是InvocationHandler 的 invoke()方法,当然,只会拦截那些你指定需要拦截的方法。
  2. 编写插件:实现Mybatis 的 Interceptor 接口并复写 intercept()方法,然后在给插件编写注解,指定要拦截哪一个接口的哪些方法即可,记住,别忘了在配置文件中配置你编写的插件。
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值