《MyBatis 理论 40 问》包含以下 2 篇文章:
MyBatis 理论 40 问(二)
- 21.如何获取生成的主键?
- [22.当实体类中的属性名和表中的字段名不一样 ,怎么办?](#22.当实体类中的属性名和表中的字段名不一样 ,怎么办?)
- [23.Mapper 编写有哪几种方式?](#23.Mapper 编写有哪几种方式?)
- [24.什么是 MyBatis 的接口绑定?有哪些实现方式?](#24.什么是 MyBatis 的接口绑定?有哪些实现方式?)
- [25.使用 MyBatis 的 mapper 接口调用时有哪些要求?](#25.使用 MyBatis 的 mapper 接口调用时有哪些要求?)
- [26.这个 Dao 接口的工作原理是什么?Dao 接口里的方法,参数不同时,方法能重载吗?](#26.这个 Dao 接口的工作原理是什么?Dao 接口里的方法,参数不同时,方法能重载吗?)
- [27.MyBatis 的 Xml 映射文件中,不同的 Xml 映射文件,id 是否可以重复?](#27.MyBatis 的 Xml 映射文件中,不同的 Xml 映射文件,id 是否可以重复?)
- [28.简述 MyBatis 的 Xml 映射文件和 MyBatis 内部数据结构之间的映射关系?](#28.简述 MyBatis 的 Xml 映射文件和 MyBatis 内部数据结构之间的映射关系?)
- [29.MyBatis 是如何将 SQL 执行结果封装为目标对象并返回的?都有哪些映射形式?](#29.MyBatis 是如何将 SQL 执行结果封装为目标对象并返回的?都有哪些映射形式?)
- [30.Xml 映射文件中,除了常见的 select | insert | updae | delete 标签之外,还有哪些标签?](#30.Xml 映射文件中,除了常见的 select | insert | updae | delete 标签之外,还有哪些标签?)
- [31.MyBatis 映射文件中,如果 A 标签通过 include 引用了 B 标签的内容,请问,B 标签能否定义在 A 标签的后面,还是说必须定义在 A 标签的前面?](#31.MyBatis 映射文件中,如果 A 标签通过 include 引用了 B 标签的内容,请问,B 标签能否定义在 A 标签的后面,还是说必须定义在 A 标签的前面?)
- [32.MyBatis 能执行一对多,一对一的联系查询吗,有哪些实现方法?](#32.MyBatis 能执行一对多,一对一的联系查询吗,有哪些实现方法?)
- [33.MyBatis 是否可以映射 Enum 枚举类?](#33.MyBatis 是否可以映射 Enum 枚举类?)
- [34.MyBatis 动态 SQL 是做什么的?都有哪些动态 SQL?能简述一下动态 SQL 的执行原理吗?](#34.MyBatis 动态 SQL 是做什么的?都有哪些动态 SQL?能简述一下动态 SQL 的执行原理吗?)
- [35.MyBatis 是如何进行分页的?分页插件的原理是什么?](#35.MyBatis 是如何进行分页的?分页插件的原理是什么?)
- [36.简述 MyBatis 的插件运行原理,以及如何编写一个插件?](#36.简述 MyBatis 的插件运行原理,以及如何编写一个插件?)
- [37.MyBatis 的一级、二级缓存](#37.MyBatis 的一级、二级缓存)
21.如何获取生成的主键?
新增标签中添加:keyProperty="ID"
即可。
xml
<insert id="insert" useGeneratedKeys="true" keyProperty="userId" >
insert into user(user_name, user_password, create_time)
values(#{userName}, #{userPassword} , #{createTime, jdbcType=TIMESTAMP})
</insert>
22.当实体类中的属性名和表中的字段名不一样 ,怎么办?
(1)通过在查询的 SQL 语句中定义字段名的别名,让字段名的别名和实体类的属性名一致。
xml
<select id="getOrder" parameterType="int" resultType="com.jourwon.pojo.Order">
select order_id id, order_no orderno, order_price price form orders
where order_id=#{id};
</select>
(2) 通过 <resultMap>
来映射字段名和实体类属性名的一一对应的关系。
xml
<select id="getOrder" parameterType="int" resultMap="orderResultMap">
select * from orders where order_id=#{id}
</select>
<resultMap type="com.jourwon.pojo.Order" id="orderResultMap">
<!-- 用id属性来映射主键字段 -->
<id property="id" column="order_id">
<!-- 用result属性来映射非主键字段,property为实体类属性名,column为数据库表中的属性 -->
<result property="orderno" column="order_no" />
<result property="price" column="order_price" />
</reslutMap>
23.Mapper 编写有哪几种方式?
(1)接口实现类继承 SqlSessionDaoSupport
:使用此种方法需要编写 mapper
接口,mapper
接口实现类、mapper.xml
文件。
- 在
sqlMapConfig.xml
中配置mapper.xml
的位置。
xml
<mappers>
<mapper resource="mapper.xml 文件的地址" />
<mapper resource="mapper.xml 文件的地址" />
</mappers>
- 定义
mapper
接口。 - 实现类集成
SqlSessionDaoSupport
。mapper
方法中可以用this.getSqlSession()
进行数据增删改查。 spring
配置。
xml
<bean id=" " class="mapper 接口的实现">
<property name="sqlSessionFactory" ref="sqlSessionFactory"></property>
</bean>
(2)使用 org.mybatis.spring.mapper.MapperFactoryBean
。
- 在
sqlMapConfig.xml
中配置mapper.xml
的位置,如果mapper.xml
和mapper
接口的名称相同且在同一个目录,这里可以不用配置。 - 定义
mapper
接口。
xml
<mappers>
<mapper resource="mapper.xml 文件的地址" />
<mapper resource="mapper.xml 文件的地址" />
</mappers>
mapper.xml
中的namespace
为mapper
接口的地址。mapper
接口中的方法名和mapper.xml
中的定义的statement
的id
保持一致。Spring
中定义。
xml
<bean id="" class="org.mybatis.spring.mapper.MapperFactoryBean">
<property name="mapperInterface" value="mapper 接口地址" />
<property name="sqlSessionFactory" ref="sqlSessionFactory" />
</bean>
(3)使用 mapper
扫描器。
mapper.xml
文件编写。mapper.xml
中的namespace
为mapper
接口的地址;mapper
接口中的方法名和mapper.xml
中的定义的statement
的id
保持一致;- 如果将
mapper.xml
和mapper
接口的名称保持一致则不用在sqlMapConfig.xml
中进行配置。
- 定义
mapper
接口。注意mapper.xml
的文件名和mapper
的接口名称保持一致,且放在同一个目录。 - 配置
mapper
扫描器。
xml
<bean class="org.mybatis.spring.mapper.MapperScannerConfigurer">
<property name="basePackage" value="mapper 接口包地址"></property>
<property name="sqlSessionFactoryBeanName" value="sqlSessionFactory"/>
</bean>
- 使用扫描器后从
spring
容器中获取mapper
的实现对象。
24.什么是 MyBatis 的接口绑定?有哪些实现方式?
接口绑定,就是在 MyBatis 中任意定义接口,然后把接口里面的方法和 SQL 语句绑定,我们直接调用接口方法就可以,这样比起原来的 SqlSession
提供的方法我们可以有更加灵活的选择和设置。
接口绑定有两种实现方式:
- 通过注解绑定,就是在接口的方法上面加上
@Select
、@Update
等注解,里面包含 SQL 语句来绑定; - 通过
xml
里面写 SQL 来绑定, 在这种情况下,要指定xml
映射文件里面的namespace
必须为接口的全路径名。当 SQL 语句比较简单时候,用注解绑定, 当 SQL 语句比较复杂时候,用xml
绑定,一般用xml
绑定的比较多。
25.使用 MyBatis 的 mapper 接口调用时有哪些要求?
- Mapper 接口方法名和
mapper.xml
中定义的每个 sql 的id
相同。 - Mapper 接口方法的输入参数类型和
mapper.xml
中定义的每个 sql 的parameterType
的类型相同。 - Mapper 接口方法的输出参数类型和
mapper.xml
中定义的每个 sql 的resultType
的类型相同。 Mapper.xml
文件中的namespace
即是 Mapper 接口的类路径。
26.这个 Dao 接口的工作原理是什么?Dao 接口里的方法,参数不同时,方法能重载吗?
- Dao 接口的工作原理是 JDK 动态代理,MyBatis 运行时会使用 JDK 动态代理为 Dao 接口生成代理
proxy
对象,代理对象proxy
会拦截接口方法,转而执行MappedStatement
所代表的 SQL,然后将 SQL 执行结果返回。 - Dao 接口里的方法是不能重载的,因为是 全限名+方法名 的保存和寻找策略。
27.MyBatis 的 Xml 映射文件中,不同的 Xml 映射文件,id 是否可以重复?
- 不同的 Xml 映射文件,如果配置了
namespace
,那么id
可以重复;如果没有配置namespace
,那么id
不能重复;毕竟namespace
不是必须的,只是最佳实践而已。 - 原因就是
namespace + id
是作为 M a p < S t r i n g , M a p p e d S t a t e m e n t > Map<String, MappedStatement> Map<String,MappedStatement> 的 Key 使用的,如果没有namespace
,就剩下id
,那么,id
重复会导致数据互相覆盖。有了namespace
,自然id
就可以重复,namespace
不同,namespace + id
自然也就不同。
28.简述 MyBatis 的 Xml 映射文件和 MyBatis 内部数据结构之间的映射关系?
MyBatis 将所有 Xml 配置信息都封装到 All-In-One 重量级对象 Configuration
内部。在 Xml 映射文件中,<parameterMap>
标签会被解析为 ParameterMap
对象,其每个子元素会被解析为 ParameterMapping
对象。 <resultMap>
标签会被解析为 ResultMap
对象,其每个子元素会被解析为 ResultMapping
对象。每一个 <select>
、<insert>
、<update>
、<delete>
标签均会被解析为 MappedStatement
对象,标签内的 SQL 会被解析为 BoundSql
对象。
29.MyBatis 是如何将 SQL 执行结果封装为目标对象并返回的?都有哪些映射形式?
- 第一种是使用
<resultMap>
标签,逐一定义列名和对象属性名之间的映射关系。 - 第二种是使用 SQL 列的别名功能,将列别名书写为对象属性名,比如
T_NAME AS NAME
,对象属性名一般是name
,小写,但是列名不区分大小写,MyBatis 会忽略列名大小写,智能找到与之对应对象属性名,你甚至可以写成T_NAME AS NaMe
,MyBatis 一样可以正常工作。
有了列名与属性名的映射关系后,MyBatis 通过反射创建对象,同时使用反射给对象的属性逐一赋值并返回,那些找不到映射关系的属性,是无法完成赋值的。
30.Xml 映射文件中,除了常见的 select | insert | updae | delete 标签之外,还有哪些标签?
还有很多其他的标签, <resultMap>
、<parameterMap>
、<sql>
、<include>
、<selectKey>
,加上动态 SQL 的 9 9 9 个标签:trim
、where
、set
、foreach
、if
、choose
、when
、otherwise
、bind
等。其中 <sql>
为 SQL 片段标签,通过 <include>
标签引入 SQL 片段, <selectKey>
为不支持自增的主键生成策略标签。
31.MyBatis 映射文件中,如果 A 标签通过 include 引用了 B 标签的内容,请问,B 标签能否定义在 A 标签的后面,还是说必须定义在 A 标签的前面?
虽然 MyBatis 解析 Xml 映射文件是按照顺序解析的,但是,被引用的 B 标签依然可以定义在任何地方,MyBatis 都可以正确识别。
原理是,MyBatis 解析 A 标签,发现 A 标签引用了 B 标签,但是 B 标签尚未解析到,尚不存在,此时,MyBatis 会将 A 标签标记为未解析状态,然后继续解析余下的标签,包含 B 标签,待所有标签解析完毕,MyBatis 会重新解析那些被标记为未解析的标签,此时再解析 A 标签时,B 标签已经存在,A 标签也就可以正常解析完成了。
32.MyBatis 能执行一对多,一对一的联系查询吗,有哪些实现方法?
能,不仅可以 一对多、一对一 ,还可以 多对多、多对一。实现方式如下:
- 单独发送一个 SQL 去查询关联对象,赋给主对象,然后返回主对象。
- 使用嵌套查询,类似 JOIN 查询,一部分是 A 对象的属性值,另一部分是关联对象 B 的属性值,好处是只要发送一个属性值,就可以把主对象和关联对象查出来。
- 子查询
33.MyBatis 是否可以映射 Enum 枚举类?
MyBatis 可以映射枚举类,不仅可以映射枚举类,MyBatis 可以映射任何对象到表的一列上。映射方式为自定义一个 TypeHandler
,实现 TypeHandler
的 setParameter()
和 getResult()
接口方法。
TypeHandler
有两个作用,一是完成从 javaType
至 jdbcType
的转换,二是完成 jdbcType
至 javaType
的转换,体现为 setParameter()
和 getResult()
两个方法,分别代表设置 sql 问号占位符参数和获取列查询结果。
34.MyBatis 动态 SQL 是做什么的?都有哪些动态 SQL?能简述一下动态 SQL 的执行原理吗?
MyBatis 动态 SQL 可以让我们在 Xml 映射文件内,以标签的形式编写动态 SQL,完成逻辑判断和动态拼接 SQL 的功能,MyBatis 提供了 9 9 9 种动态 SQL 标签:trim
、where
、set
、foreach
、if
、choose
、when
、otherwise
、bind
。
其执行原理为,使用 OGNL
从 SQL 参数对象中计算表达式的值,根据表达式的值动态拼接 SQL,以此来完成动态 SQL 的功能。
35.MyBatis 是如何进行分页的?分页插件的原理是什么?
MyBatis 使用 RowBounds
对象进行分页,它是针对 ResultSet
结果集执行的内存分页,而非物理分页,可以在 SQL 内直接书写带有物理分页的参数来完成物理分页功能,也可以使用分页插件来完成物理分页。
分页插件的基本原理是使用 MyBatis 提供的插件接口,实现自定义插件,在插件的拦截方法内拦截待执行的 SQL,然后重写 SQL,根据 dialect
方言,添加对应的物理分页语句和物理分页参数。
举例:
sql
select * from student
拦截 SQL 后重写为:
sql
select t.* from (select * from student) t limit 0, 10
36.简述 MyBatis 的插件运行原理,以及如何编写一个插件?
MyBatis 仅可以编写针对 ParameterHandler
、ResultSetHandler
、StatementHandler
、Executor
这 4 4 4 种接口的插件,MyBatis 使用 JDK 的动态代理,为需要拦截的接口生成代理对象以实现接口方法拦截功能,每当执行这 4 4 4 种接口对象的方法时,就会进入拦截方法,具体就是 InvocationHandler
的 invoke()
方法,当然,只会拦截那些你指定需要拦截的方法。
实现 MyBatis 的 Interceptor
接口并复写 intercept()
方法,然后在给插件编写注解,指定要拦截哪一个接口的哪些方法即可,记住,别忘了在配置文件中配置你编写的插件。
37.MyBatis 的一级、二级缓存
- 一级缓存:基于
PerpetualCache
的 HashMap 本地缓存,其存储作用域为 Session,当 Sessionflush
或close
之后,该 Session 中的所有 Cache 就将清空,默认打开一级缓存。 - 二级缓存与一级缓存其机制相同,默认也是采用
PerpetualCache
,HashMap 存储,不同在于其存储作用域为Mapper
(Namespace
),并且可自定义存储源,如Ehcache
。默认不打开二级缓存,要开启二级缓存,使用二级缓存属性类需要实现Serializable
序列化接口(可用来保存对象的状态),可在它的映射文件中配置<cache/>
。 - 对于缓存数据更新机制,当某一个作用域(一级缓存 Session / 二级缓存 Namespace)进行了
C
/U
/D
操作后,默认该作用域下所有select
中的缓存将被clear
。