MyBatis面试题
20、MyBatis实现一对一有几种方式?具体怎么操作的?
有联合查询和嵌套查询,联合查询是几个表联合查询,只查询一次,通过在resultMap里面配置 association节点配置一对一的类就可以完成;
嵌套查询是先查一个表,根据这个表里面的结果的外键 id,去再另外一个表里面查询数据,也是通过 association配置,但另外一个表的查询通过 select属性配置。
21、MyBatis实现一对多有几种方式,怎么操作的?
有联合查询和嵌套查询。联合查询是几个表联合查询,只查询一次,通过在resultMap里面的collection节点配置一对多的类就可以完成;嵌套查询是先查一个表,根据这个表里面的结果的外键 id,去再另外一个表里面查询数据,也是通过配置 collection,但另外一个表的查询通过 select节点配置。
22、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,支持延迟加载的原理都是一样的。
23、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。
24、什么是 MyBatis的接口绑定?有哪些实现方式?
接口绑定,就是在 MyBatis中任意定义接口,然后把接口里面的方法和 SQL语句绑定,我们直接调用接口方法就可以,这样比起原来了 SqlSession提供的方法我们可以有更加灵活的选择和设置。
接口绑定有两种实现方式,一种是通过注解绑定,就是在接口的方法上面加上
@Select、@Update等注解,里面包含 Sql语句来绑定;另外一种就是通过 xml里面写 SQL来绑定,在这种情况下,要指定 xml映射文件里面的 namespace必须为接口的全路径名。当 Sql语句比较简单时候,用注解绑定,当 SQL语句比较复杂时候,用 xml绑定,一般用 xml绑定的比较多。
25、使用 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接口的类路径。