Mybatis经常使用知识点总结

1. #{}和${}的区别是什么?
  ${}是Properties文件中的变量占位符,它能够用于标签属性值和sql内部,属于静态文本替换,好比${driver}会被静态替换为com.mysql.jdbc.Driver。
  #{}是sql的参数占位符,Mybatis会将sql中的#{}替换为?号,在sql执行前会使用PreparedStatement的参数设置方法,按序给sql的?号占位符设置参数值,好比ps.setInt(0, parameterValue)。
  后者能够防止SQL注入,前者不能够。html

 

2. Xml映射文件中,除了常见的select|insert|updae|delete标签以外,还有哪些?
  还不少其余的标签,<resultMap>、<parameterMap>、<sql>、<include>、<selectKey>,加上动态sql的9个标签,trim|where|set|foreach|if|choose|when|otherwise|bind等。java

 

3. 最佳实践中,一般一个Xml映射文件,都会写一个Dao接口与之对应,这个Dao接口的工做原理是什么?Dao接口里的方法,参数不一样时,方法能重载吗?
  Dao接口,就是人们常说的Mapper接口,接口的全限名,就是映射文件中的namespace的值,接口的方法名,就是映射文件中MappedStatement的id值,接口方法内的参数,就是传递给sql的参数。Mapper接口是没有实现类的,当调用接口方法时,接口全限名+方法名拼接字符串做为key值,可惟必定位一个MappedStatement。在Mybatis中,每个<select>、<insert>、<update>、<delete>标签,都会被解析为一个MappedStatement对象。
  Dao接口里的方法,是不能重载的,由于是全限名+方法名的保存和寻找策略。
  Dao接口的工做原理是JDK动态代理,Mybatis运行时会使用JDK动态代理为Dao接口生成代理proxy对象,代理对象proxy会拦截接口方法,转而执行MappedStatement所表明的sql,而后将sql执行结果返回。mysql

 

4. 简述Mybatis的插件运行原理,以及如何编写一个插件。
  Mybatis仅能够编写针对ParameterHandler、ResultSetHandler、StatementHandler、Executor这4种接口的插件,Mybatis使用JDK的动态代理,为须要拦截的接口生成代理对象以实现接口方法拦截功能,每当执行这4种接口对象的方法时,就会进入拦截方法,具体就是InvocationHandler的invoke()方法,固然,只会拦截那些你指定须要拦截的方法。
  实现Mybatis的Interceptor接口并复写intercept()方法,而后在给插件编写注解,指定要拦截哪个接口的哪些方法便可,记住,别忘了在配置文件中配置你编写的插件。sql

 

5. Mybatis是如何将sql执行结果封装为目标对象并返回的?都有哪些映射形式?
  第一种是使用<resultMap>标签,逐必定义列名和对象属性名之间的映射关系。
  第二种是使用sql列的别名功能,将列别名书写为对象属性名。缓存

 

6. 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

 

7. Mybatis的Xml映射文件中,不一样的Xml映射文件,id是否能够重复?
  不一样的Xml映射文件,若是配置了namespace,那么id能够重复;若是没有配置namespace,那么id不能重复;毕竟namespace不是必须的,只是最佳实践而已。
  缘由就是namespace+id是做为Map<String, MappedStatement>的key使用的,若是没有namespace,就剩下id,那么,id重复会致使数据互相覆盖。有了namespace,天然id就能够重复,namespace不一样,namespace+id天然也就不一样。app

 

8. Mybatis都有哪些Executor执行器?它们之间的区别是什么?
  Mybatis有三种基本的Executor执行器,SimpleExecutor、ReuseExecutor、BatchExecutor。
  SimpleExecutor:每执行一次update或select,就开启一个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生命周期范围内。ide

 

9. Mybatis是否能够映射Enum枚举类?
  Mybatis能够映射枚举类,不单能够映射枚举类,Mybatis能够映射任何对象到表的一列上。映射方式为自定义一个TypeHandler,实现TypeHandler的setParameter()和getResult()接口方法。
  TypeHandler有两个做用,一是完成从javaType至jdbcType的转换,二是完成jdbcType至javaType的转换,体现为setParameter()和getResult()两个方法,分别表明设置sql问号占位符参数和获取列查询结果。spa

 

10. Mybatis映射文件中,若是A标签经过include引用了B标签的内容,请问,B标签可否定义在A标签的后面,仍是说必须定义在A标签的前面?
  虽然Mybatis解析Xml映射文件是按照顺序解析的,可是,被引用的B标签依然能够定义在任何地方,Mybatis均可以正确识别。
  原理是: Mybatis解析A标签,发现A标签引用了B标签,可是B标签还没有解析到,尚不存在,此时,Mybatis会将A标签标记为未解析状态,而后继续解析余下的标签,包含B标签,待全部标签解析完毕,Mybatis会从新解析那些被标记为未解析的标签,此时再解析A标签时,B标签已经存在,A标签也就能够正常解析完成了。插件

 

11. mybatis 中的trim有什么做用?

  trim 的prefixOverrides 属性会忽略经过管道分隔的文本序列。它带来的结果就是全部在 prefixOverrides 属性中指定的内容将被移除,而且插入 prefix 属性中指定的内容。

  以下示例:

  select * from user 

  <trim prefix="WHERE" prefixoverride="AND |OR">

    <if test="id != null"> AND id=#{id}</if>

    <if test="name!= null"> AND name=#{name}</if>

  </trim>

  id和name的值都不为null时对应的SQL为:select * from user where  id = 'xx' and name = 'xx'

  上面两个属性的意思以下:

  prefix:前缀      

  prefixoverride:去掉第一个and或者是or


   再看以下示例:

update user
 
  <trim prefix="set" suffixoverride="," suffix=" where id = #{id} ">
 
    <if test="name != null"> name=#{name} , </if>
 
    <if test="age != null"> age=#{age} ,  </if>
 
  </trim>
 
  name和age都不为null时对应的SQL为:update user set name='xx' , age='xx' where id='x'
 
上面三个属性的意义以下:
   
prefix:前缀
 
suffixoverride:去掉最后一个逗号(也能够是其余的标记,就像是上面前缀中的and同样)
 
suffix:后缀

  

12. MyBatis中命名空间(namespace)的做用?  在大型项目中,可能存在大量的SQL语句,这时候为每一个SQL语句起一个惟一的标识(ID)就变得并不容易了。为了解决这个问题,在MyBatis中,能够为每一个映射文件起一个惟一的命名空间,这样定义在这个映射文件中的每一个SQL语句就成了定义在这个命名空间中的一个ID。只要咱们可以保证每一个命名空间中这个ID是惟一的,即便在不一样映射文件中的语句ID相同,也不会再产生冲突了。

相关文章
相关标签/搜索