Mybatis面试题二

  • 15、Mybatis动态sql有什么用?执行原理?有哪些动态sql?

  • Mybatis动态sql可以在Xml映射文件内,以标签的形式编写动态sql,执行原理是根据表达式的值 完成逻辑判断并动态拼接sql的功能。

  • Mybatis提供了9种动态sql标签:

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

  • 16、Xml映射文件中,除了常见的select|insert|updae|delete标签之外,还有哪些标签?

  • 答:、、、、,加上动态sql的9个标签,其中 为sql片段标签,通过 标签引入sql片段, 为不支持自增的主键生成策略标签。

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

  • 不同的Xml映射文件,如果配置了namespace,那么id可以重复;如果没有配置namespace,那么id不能重复;

  • 原因就是namespace+id是作为Map <String,MapperStatement>的key使用的,如果没有namespace,就剩下id,那么,id重复会导致数据互相覆盖。有了namespace,自然id就可以重复,namespace不同,namespace+id自然也就不同。

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

  • Hibernate属于全自动ORM映射工具,使用Hibernate查询关联对象或者关联集合对象时,可以根据对象关系模型直接获取,所以它是全自动的。而Mybatis在查询关联对象或关联集合对象时,需要手动编写sql来完成,所以,称之为半自动ORM映射工具。

  • 19、 一对一、一对多的关联查询 ?

 <mapper namespace="com.lcb.mapping.userMapper">  
-     <!--association  一对一关联查询 -->  
-     <select id="getClass" parameterType="int" resultMap="ClassesResultMap">  
-         select * from class c,teacher t where c.teacher_id=t.t_id and c.c_id=#{id}
-     </select>  
-     <resultMap type="com.lcb.user.Classes" id="ClassesResultMap">  
-         <!-- 实体类的字段名和数据表的字段名映射 -->  
-         <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>  
-     </resultMap>  
-     <!--collection  一对多关联查询 -->  
-     <select id="getClass2" parameterType="int" resultMap="ClassesResultMap2">  
-         select * from class c,teacher t,student s where c.teacher_id=t.t_id and c.c_id=s.class_id and c.c_id=#{id}
-     </select>  
-     <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>  
- </mapper>
  • 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序列化接口(可用来保存对象的状态),可在它的映射文件中配置 ;

  • 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接口的类路径。

  • 26、Mapper编写有哪几种方式?

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

  • 1、在sqlMapConfig.xml中配置mapper.xml的位置

- <mappers>
-     <mapper resource="mapper.xml文件的地址" />
-     <mapper resource="mapper.xml文件的地址" />
- </mappers>
  • 2、定义mapper接口

  • 3、实现类集成SqlSessionDaoSupportmapper方法中可以this.getSqlSession()进行数据增删改查。

  • 4、spring 配置

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

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

- <mappers>
-     <mapper resource="mapper.xml文件的地址" />
-     <mapper resource="mapper.xml文件的地址" />
- </mappers>
  • 2、定义mapper接口:

  • 1、mapper.xml中的namespace为mapper接口的地址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" /> 
  • 第三种:使用mapper扫描器:

  • 1、mapper.xml文件编写:

  • mapper.xml中的namespace为mapper接口的地址;mapper接口中的方法名和mapper.xml中的定义的statement的id保持一致;如果将mapper.xml和mapper接口的名称保持一致则不用在sqlMapConfig.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"/> 
  • 4、使用扫描器后从spring容器中获取mapper的实现对象。
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值