mybatis-usage

92 阅读3分钟

mybatis-usage

interview

半ORM框架

  • 自己写原生的sql
  • 全自动: gorm直接操作对象User.findone('')

#{}和${}的区别

#{}是预编译处理,PreparedStatement; ${}是字符串替换

dao和xml文件的关系

Dao接口即 Mapper接口。接口的全限名,就是映射文件中mapper.namespace; 接口的方法名,就是映射文件中 Mapper 的 Statement 的 id 值;接口方法内 的参数,就是传递给 sql 的参数;

Mapper接口的工作原理是 JDK 动态代理;生成代理Proxy; 代理的任务是: 代理执行指定的MapperStatement, key=>class.method(namespace+id)

Mapper接口是没有实现类的,当调用接口方法时,接口全限名+方法名拼接字符串作为key值,可唯一定位一个MapperStatement

ResultMap与Java对象的映射关系

  1. resultMap
  2. sql列的别名

Mybatis 通过反射创建对象,同时使用反射给对象的属性逐一赋值并返回

动态sql

trim | where | set | foreach | if | choose| when | otherwise | bind

标签

  • select|insert|updae|delete
  • 、、
  • has-a JAVA对象 bean
  • has-a 容器 List
class A{
  B b;  // association
  List<B> bs; //collection
}
class B{}

1对多查询:

<resultMap type="com.lcb.user.Classes" id="ClassesResultMap2">
  <id property="id" column="c_id"/>
  <result property="name" column="c_name"/>
  <association property="teacher" javaType="com.lcb.user.Teacher">
     <id property="id" column="t_id"/>
     <result property="name" column="t_name"/>
  </association>
  <collection property="student" ofType="com.lcb.user.Student">
     <id property="id" column="s_id"/>
     <result property="name" column="s_name"/>
  </collection>
</resultMap>

嵌套查询:

column="{subId=id,subName=userName}"; key是子查询中的入参;value是外层查询的结果字段:property;

嵌套查询语句


// 父查询
<resultMap>
   <result property="parentId" column="parent_id"/>
   <collection property="Java属性名" ofType="Java类名" javaType="ArrayList" column="{subId=parentId}关联子查询"  select="sub_select_id(自查询Id)">
   </ccollection>
</resultMap>

// 子查询
<select parameterType="int(子查询的入参)" resultType="子查询的Java类名" id="sub_select_id(子查询的id)">
     select * from sub where id = #{subId}
<select>

嵌套查询结果

// 父查询
<resultMap>
    <result property="parentId" column="parent_id"/>
    <collection property="Java属性名" ofType="另一Java类名" javaType="ArrayList" resultMap="sub_result_map(子结果映射)"/>
</resultMap>

// 子结果
<resultMap="sub_result_map" type="子Java类名:subClass">
    <id property="id" column=""/>
</resultMap>

延迟加载

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

二级缓存

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

=> cache-aside方法....

接口绑定的实现

  1. @Select、@Update 等注解
  2. xml文件

自定义插件

ParameterHandler、ResultSetHandler、 StatementHandler、Executor 这 4 种接口的插件;

通过jdk代理执行InvocationHandler的invoke()方法

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

涉及到的设计模式:

  1. 调用连模式加载自定义的Interceptor
  2. jdk动态代理执行插件:InvocationHandler
  3. 通过注解匹配接口和方法; => 类似策略;通过注解实现

自定义插件

ps

  1. www.cnblogs.com/longxok/p/1…