2024-07-12
한어Русский языкEnglishFrançaisIndonesianSanskrit日本語DeutschPortuguêsΕλληνικάespañolItalianoSuomalainenLatina
《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) 通过 <resultMap>
来映射字段名和实体类属性名的一一对应的关系。
<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>
(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
>
Map<String, MappedStatement>
Map<String,MappedStatement> 的 Key 使用的,如果没有 namespace
,就剩下 id
,那么,id
重复会导致数据互相覆盖。有了 namespace
,自然 id
就可以重复,namespace
不同,namespace + id
自然也就不同。MyBatis 将所有 Xml 配置信息都封装到 All-In-One 重量级对象 Configuration
内部。在 Xml 映射文件中,<parameterMap>
标签会被解析为 ParameterMap
对象,其每个子元素会被解析为 ParameterMapping
对象。 <resultMap>
标签会被解析为 ResultMap
对象,其每个子元素会被解析为 ResultMapping
对象。每一个 <select>
、<insert>
、<update>
、<delete>
标签均会被解析为 MappedStatement
对象,标签内的 SQL 会被解析为 BoundSql
对象。
<resultMap>
标签,逐一定义列名和对象属性名之间的映射关系。T_NAME AS NAME
,对象属性名一般是 name
,小写,但是列名不区分大小写,MyBatis 会忽略列名大小写,智能找到与之对应对象属性名,你甚至可以写成 T_NAME AS NaMe
,MyBatis 一样可以正常工作。有了列名与属性名的映射关系后,MyBatis 通过反射创建对象,同时使用反射给对象的属性逐一赋值并返回,那些找不到映射关系的属性,是无法完成赋值的。
还有很多其他的标签, <resultMap>
、<parameterMap>
、<sql>
、<include>
、<selectKey>
,加上动态 SQL 的
9
9
9 个标签:trim
、where
、set
、foreach
、if
、choose
、when
、otherwise
、bind
等。其中 <sql>
为 SQL 片段标签,通过 <include>
标签引入 SQL 片段, <selectKey>
为不支持自增的主键生成策略标签。
虽然 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
序列化接口(可用来保存对象的状态),可在它的映射文件中配置 <cache/>
。C
/ U
/ D
操作后,默认该作用域下所有 select
中的缓存将被 clear
。