zoukankan      html  css  js  c++  java
  • mybatis面试题

    转载: https://www.cnblogs.com/qmillet/p/12523636.html

    1. 最佳实践中,通常一个Xml映射文件,都会写一个Dao接口与之对应,请问,这个Dao接口的工作原理是什么?Dao接口里的方法,参数不同时,方法能重载吗?

    xml对应Dao接口: 是由于xml定义了命名空间是Dao接口路径,所以相互对应.

    可以一个Dao接口对应多个xml文件,但是标签id不能重复,假如重复就会出现执行SQL混乱的情况

    Dao接口的工作原理是JDK动态代理,Mybatis运行时会使用JDK动态代理为Dao接口生成代理proxy对象,代理对象proxy会拦截接口方法,转而执行MappedStatement所代表的SQL,然后将SQL执行结果返回.

    参数不同时,不能重载. 是由于是使用类路径+方法名确定唯一性的,重载相当于出现重复

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

    Mybatis使用RowBounds对象进行分页,它是针对ResultSet结果集执行的内存分页,而非物理分页,可以在SQL内直接书写带有物理分页的参数来完成物理分页功能,也可以使用分页插件来完成物理分页.

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

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

      Mybatis仅可以编写针对ParameterHandler,ResultHandler,StatementHandler,Executor这四种接口的插件,Mybatis使用JDK的动态代理,为需要拦截的接口生成代理对象以实现接口方法拦截功能,每当执行这4种接口对象的方法时,就会进入拦截方法,具体就是InvocationHandler的invoke()方法,当然,只会拦截那些你指定需要拦截的方法

    实现Mybatis的Interceptor接口并复写interceptor()方法,然后在给插件编写注解,指定要拦截哪一个接口的哪些方法即可,记住,别忘了在配置文件中配置你编写的插件

    4、Mybatis执行批量插入,能返回数据库主键列表吗?

    能,JDBC都能,Mybatis当然也能.

    mybatis批量导入返回id列表需要注意以下几点

    1. 确保 mybatis 版本在 3.3.1 以上
    2. batchInsert 方法上不能加 @param()
    3. batchInsert 方法只能一个参数
    4. batchInsert 返回值为 Integer 或 void,不能写 List
    5. 如果你的自增id数据库字段和实体类属性不一致,如 user_id 和 userId, 需要写成useGeneratedKeys="true" keyColumn="user_id" keyProperty="userId"

    5、Mybatis动态sql是做什么的?都有哪些动态sql?能简述一下动态sql的执行原理不?

      Mybatis动态SQL可以让我们在XML映射文件内,以标签的形式编写动态SQL,完成逻辑判断和动态拼接SQL的功能,Mybatis提供了9种动态SQL标签trim|where|set|foreach|if|choose|when|otherwise|bind

    其执行原理为,使用OGNL从SQL参数对象中计算表达是的值,根据表达式的值动态拼接SQL,以此来完成动态SQL的功能

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

      第一种是使用<resultMap>标签,逐一定义列名和对象名之前的映射关系.第二种是使用SQL列的别名功能,将列别名书写为对象属性名,比如T_NAME AS NAME,对象属性名一般是name,小写,但是列名不区分大小写,Mybatis会忽略列名大小写,智能找到与之对应对象属性名,你甚至可以写成T_NAME AS NAME,Mybatis一样可以正常工作.

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

      以查询为例:

      入口类: PreparedStatementHandler 经过方法query(Statement statement, ResultHandler resultHandler)

      经过DefaultResultSetHandler的一系列方法

      最终在DefaultResutlSetHandler中方法applyPropertyMappings(ResultSetWrapper rsw, ResultMap resultMap, MetaObj metaObject, ResultLoaderMap lazyLoader, String columnPrefix) 完成自动映射

    7、Mybatis能执行一对一、一对多的关联查询吗?都有哪些实现方式,以及它们之间的区别。

      能, Mybatis不仅可以执行一对一,一对多的关联查询,还可以执行多对一,多对多的关联查询,多对一查询,其实就是一对一查询,

    只需要 吧selectOne()修改为selectList既可.

      关联对象查询,有两种实现方式,一种是单独发送一个sql去查询关联对象,赋给主对象,然后返回主对象.

      另一种是使用嵌套查询,嵌套查询的含义为使用join查询,一部分列是A对象的属性值,另外一部分列是关联对象B的属性值,

      好处是只发一个SQL查询,就可以把主对象和其关联查询对象查出来.

      那么问题来了,join查询出来100条记录,如何确定主对象是5个,而不是100个?其去重复的原理是<resultMap>标签内的<id>

      子标签,指定了唯一确定一条记录的id列,Mybatis根据<id>列值来完成100条记录的去重复功能,<id>可以有多个,代表了联合主键的语义.

      同样主对象的关联对象,也是根据这个原理去重复的,尽管一般情况下,只有主对象会出现有重复记录,关联对象一般不会重复.

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

    9、Mybatis的Xml映射文件中,不同的Xml映射文件,id是否可以重复?

      不同的XML映射文件中,如果配置了namespace,那么id可以重复;如果没有配置namespace,那么id不能重复;毕竟namespace不是必须的,只是最佳实践而已.

      原因就是namespace+id是作为Map<String,MappedStatement>的key使用的,如果没有namespace,就剩下id,

    那么,id重复会导致数据互相覆盖.有了namespace,自然id就可以重复,namespace不同,namespace+id自然也就不同.

    10、Mybatis中如何执行批处理?

      使用BatchExecutor完成批处理

    11、Mybatis都有哪些Executor执行器?它们之间的区别是什么?

      Mybatis有三种基本的Executor执行器,SimpleExecutor,ReuseExecutor,BatchExecutor.

      SimpleExecutor: 每执行一次update或select,以SQL作为key查找Statement对象,存在就使用,不存在就创建,用完后,立即关闭Statement对象,

      

      ReuseExecutor: 执行update或select,以SQL作为key查找Statement对象,存在就使用,不存在就创建,用完后,不关闭Statement对象,

      而是放置于Map<String,Statement>内,供下一次使用.简言之,就是重复使用Statement对象

      BatchExecutor: 执行update(没有select,JDBC批处理不支持select),将所有SQL都添加到批处理中(addBatch()),

    等待统一执行(executeBatch()),它缓存了多个Statement对象,每个Statement对象都是addBatch()完毕后,

    等待逐一执行(executeBatch())批处理.与JDBC批处理相同.

    作用范围: Executor的这些特点,都严格限制在SQLSession生命周期范围内.

    12、Mybatis中如何指定使用哪一种Executor执行器?

      在Mybatis配置文件中,可以指定默认的ExecutorType执行器类型,

      也可以手动给DefaultSqlSessionFactory的创建SqlSession的方法传递ExecutorType类型参数.

    13、Mybatis是否可以映射Enum枚举类?

      Mybatis可以映射枚举类,不单可以映射枚举类,Mybatis可以映射任何对象到表的一列上.映射方式为自定义一个TypeHandler,

      实现TypeHandler的setParameter()和getResult()接口方法.TypeHandler有两个作用,

      一是完成从javaType至jabcType的转换,二是完成jdbcType和javaType的转换,提现为setParameter()和getResult()两个方法,

      分别代表设置SQL问号占位符参数和活回去列查询结果

    14、Mybatis映射文件中,如果A标签通过include引用了B标签的内容,请问,B标签能否定义在A标签的后面,还是说必须定义在A标签的前面?

      虽然Mybatis解析xml映射文件是按照顺序解析的,但是,被引用的B标签依然可以定义在任何地方,Mybatis都可以识别.

      原理是,Mybatis解析A标签,发现A标签引用了B标签,但是B标签尚未解析到,尚不存在,此时,Mybatis会将A标签标记为未解析状态,

      然后继续解析余下的标签,包含B标签,待所有标签解析完毕,Mybatis会重新解析哪些被标记为解析的标签,此时在解析A标签时,

      b标签已经存在,A标签也就可以正常解析完成了.

    15、简述Mybatis的Xml映射文件和Mybatis内部数据结构之间的映射关系?

      Mybatis将所有Xml配置信息都封装到All-In-One重量级对象Configuration内部.在Xml映射文件中,<parameterMap>标签会被解析为

      ParameterMap对象,其每个子元素会被解析为ParameterMapping对象.

      <resultMap>标签会被解析为ResultMap对象,其每个子元素会被解析为ResultMapping对象.每一个<Select>,<insert>,<update>,

      <delete>标签均会被解析为MappedStatement对象,标签内的SQL会被解析为Bounds对象.

    16、为什么说Mybatis是半自动ORM映射工具?它与全自动的区别在哪里?

      Hibernate属于全自动ORM映射工具,使用Hibernate查询关联对象或者关联集合对象时,可以根据对象关系模型直接获取,所以它是全自动的.

      而Mybatis在查询关联对象或关联集合对象时,需要手动编写SQL来完成.所以,称之为半自动ORM映射工具.

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

      1.使用别名

      2.使用<resultMap>配置映射关系

    18. 在mapper中如何传递多个参数?

      1.使用占位符的思想使用#{0},#{1}

      2.使用@Param注解方式定义参数名

      #{username}

      3.对象形式,传递使用属性名,如果是map的话,使用key

    19. Mybatis动态sql是做什么的?都有哪些动态sql?能简述一下动态sql的执行原理不?

      Mybatis动态sql可以让我们在xml映射文件内,以标签的形式编写动态SQL,完成逻辑判断和动态拼接SQL功能

      Mybatis提供了9中动态SQL标签:trim|where|set|foreach|if|choose|when|otherwise|bind

      其执行原理为,使用OGNL从SQL参数对象中计算表达式的值,根据表达式的值动态拼接SQL,以此来完成动态SQL的功能

    20. Mybatis的Xml映射文件中,不同的Xml映射文件,id是否可以重复?

      如果配置了namespace那么当然是可以重复的,因为我们的statement实际上就是namespace+id

      如果没有配置namespace的话,那么相同的id就导致覆盖了.

    21. 接口绑定有几种实现方式,分别是怎么实现的?

      接口绑定有两种实现方式:

      1.是通过注解绑定,就是在接口的方法上面加上@Select@Update等注解里面包含SQL语句来绑定

      2.是通过xml里面写SQL来绑定,在这种情况下,要指定xml映射文件里面的namespace必须为接口的全路径名.

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

      mybatis使用RowBounds对象进行分页,它是针对ResultSet结果集执行的内存分页,而非物理分页,在SQL内直接书写物理分页

      的参数来完成物理分页功能,也可以使用分页插件来完成物理分页.

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

    24. 简述Mybatis的插件运行原理,以及如何编写一个插件

      mybatis仅可以编写针对ParamenterHandler,ResultSetHandler,Statementhandler,Executor这4种接口的插件,Mybatis使用JDK

      的动态代理,为需要拦截的接口生成代理对象以实现接口方法拦截功能,每当执行这4种接口对象的方法时,就会进入拦截方法,

      具体就是InvocationHandler的invoke()方法,当然,只会拦截那些你指定需要拦截的方法.

      实现mybatis的Interceptor接口并复写intercept()方法,然后在给插件编写注解,指定要拦截哪一个接口的哪些方法即可,

      记住,别忘了在配置文件中配置你编写的插件.

  • 相关阅读:
    服务端获取实际IP工具类
    vue-element-admin整合服务端代理api
    elemengui分页
    vue获取浏览器地址栏参数
    el-table-column动态判断显示性别男女
    nginx代理静态页面添加二级目录
    nginx配置代理指向Redis
    SpringBoot整合Redis日志反复提示Redis重连问题
    centos7安装Redis
    nohup启动jar包
  • 原文地址:https://www.cnblogs.com/yxgmagic/p/15657814.html
Copyright © 2011-2022 走看看