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