Mybatis第二天

Mybatis第二天
1、高级结果映射(一对一、一对多、多对多)(重点)
2、延迟加载
3、查询缓存
4、Spring和mybatis的整合(重点)
5、逆向工程
2高级结果映射
2.1数据模型分析
1、明确每张表存储的信息
2、明确每张表中关键字段(主键、外键、非空)
3、明确数据库中表与表之间的外键关系
4、明确业务中表与表的关系(建立在具体的业务)
2.2一对一映射
2.2.1需求
查询订单信息,关联查询用户信息

2.2.2Sql
主信息:orders
从信息:user

SELECT
orders.id,
orders.user_id,
orders.number,
user.username,
user.sex
FROM
orders,
USER
WHERE orders.user_id = user.id

2.2.3resultType
2.2.3.1创建扩展类

2.2.3.2映射文件

2.2.3.3Mapper接口

2.2.3.4测试代码

2.2.3.5小结
使用resultType来进行一对一结果映射,查询出的列的个数和映射的属性的个数要一致。而且映射的属性要存在与一个大的对象中,它是一种平铺式的映射,即数据库查询出多少条记录,则映射成多少个对象。

2.2.4resultMap
使用resultMap来进行一对一结果映射,它是将关联对象添加到主信息的对象中,具体说是对象嵌套对象的一种映射方式。
2.2.4.1修改扩展类

2.2.4.2映射文件

2.2.4.3Mapper接口

2.2.4.4测试代码

2.2.5小结
在一对一结果映射时,使用resultType更加简单方便,如果有特殊要求(对象嵌套对象)时,需要使用resultMap进行映射,比如:查询订单列表,然后在点击列表中的查看订单明细按钮,这个时候就需要使用resultMap进行结果映射。而resultType更适应于查询明细信息,比如,查询订单明细列表。

2.3一对多映射
2.3.1需求
查询订单信息,关联查询订单明细信息及用户信息

2.3.2Sql
主信息:orders
从信息:orderdetail、user

SELECT
orders.id,
orders.user_id,
orders.number,
user.username,
user.sex,
orderdetail.id detailId,
orderdetail.items_id,
orderdetail.items_num
FROM
orders,
USER,
orderdetail
WHERE orders.user_id = user.id
AND orders.id = orderdetail.orders_id
2.3.3修改扩展类

2.3.4映射文件

2.3.5Mapper接口

2.3.6测试代码

2.4多对多映射
多对多映射是一对多映射的特例

2.4.1需求
查询用户信息,关联查询该用户购买的商品信息

2.4.2Sql
主信息:user
从信息:items、orders、orderdetail

SELECT
orders.id,
orders.user_id,
orders.number,
user.username,
user.sex,
orderdetail.id detailId,
orderdetail.items_id,
orderdetail.items_num,
items.name,
items.price
FROM
orders,
USER,
orderdetail,
items
WHERE orders.user_id = user.id
AND orders.id = orderdetail.orders_id
AND orderdetail.items_id = items.id

2.4.3修改po类
在User类中添加List orders;

在Orders类中添加List detailList;

在Orderdetail中添加Items items;

2.4.4Mapper接口

2.4.5映射文件

2.4.6测试代码

3延迟加载
3.1什么是延迟加载
延迟加载又叫懒加载,也叫按需加载。也就是说先加载主信息,在需要的时候,再去加载从信息。

在mybatis中,resultMap标签 的association标签和collection标签具有延迟加载的功能。

3.2需求
查询订单信息,关联查询用户信息
1、创建一个statement来查询订单信息
2、创建一个statement来查询用户信息

3.3映射文件
1、创建查询订单信息的映射文件

2、创建查询用户信息的映射文件

3.4Mapper接口

3.5测试代码

3.6设置延迟加载
在SqlMapConfig.xml中,配置settings标签

4查询缓存
4.1Mybatis的缓存理解

Mybatis的缓存,包括一级缓存和二级缓存

一级缓存指的就是sqlsession,在sqlsession中有一个数据区域,是map结构,这个区域就是一级缓存区域。一级缓存中的key是由sql语句、条件、statement等信息组成一个唯一值。一级缓存中的value,就是查询出的结果对象。

二级缓存指的就是同一个namespace下的mapper,二级缓存中,也有一个map结构,这个区域就是一级缓存区域。一级缓存中的key是由sql语句、条件、statement等信息组成一个唯一值。一级缓存中的value,就是查询出的结果对象。

一级缓存是默认使用的。
二级缓存需要手动开启。

4.2一级缓存
4.2.1原理

4.2.2测试1

4.2.3测试2

4.3二级缓存
4.3.1原理

4.3.2开启二级缓存
1、开启二级缓存的总开关

2、在mapper映射文件中开启二级缓存

4.3.3序列化

4.3.4测试1

4.3.5测试2

4.3.6禁用缓存
默认值是true

4.3.7刷新缓存

4.3.8整合ehcache
Mybatis本身是一个持久层框架,它不是专门的缓存框架,所以它对缓存的实现不够好,不能支持分布式。

Ehcache是一个分布式的缓存框架。

4.3.8.1什么是分布式
系统为了提高性能,通常会对系统采用分布式部署(集群部署方式)

4.3.8.2整合思路
Cache是一个接口,它的默认实现是mybatis的PerpetualCache。如果想整合mybatis的二级缓存,那么实现Cache接口即可。

4.3.8.3添加jar包

4.3.8.4设置映射文件中cache标签的type值为ehcache的实现类

4.3.8.5添加ehcache的配置文件
在config下,创建ehcache.xml

4.3.8.6测试ehcache的二级缓存

4.3.9应用场景
使用场景:对于访问响应速度要求高,但是实时性不高的查询,可以采用二级缓存技术。
注意:在使用二级缓存的时候,要设置一下刷新间隔(cache标签中有一个flashInterval属性)来定时刷新二级缓存,这个刷新间隔根据具体需求来设置,比如设置30分钟、60分钟等,单位为毫秒。
4.3.10局限性
Mybatis二级缓存对细粒度的数据,缓存实现不好。
场景:对商品信息进行缓存,由于商品信息查询访问量大,但是要求用户每次查询都是最新的商品信息,此时如果使用二级缓存,就无法实现当一个商品发生变化只刷新该商品的缓存信息而不刷新其他商品缓存信息,因为二级缓存是mapper级别的,当一个商品的信息发送更新,所有的商品信息缓存数据都会清空。
解决此类问题,需要在业务层根据需要对数据有针对性的缓存。
比如可以对经常变化的 数据操作单独放到另一个namespace的mapper中。
5Mybatis整合spring
5.1整合思路
1、数据源信息交给spring管理
2、SqlSessionFactory交给spring进行单例管理
3、由spring来管理原始dao的实现类或者mapper代理的代理类。

5.2需求
使用原始dao方式和mapper代理方式实现以下功能:
根据用户ID查询商品信息
5.3工程搭建
Mysql的驱动包
Mybatis的核心包和依赖包
Mybatis和spring的整合包
Spring的包
dbcp数据库连接池包

5.4具体整合
5.4.1整合配置文件
5.4.1.1Mybatis
在config下,创建mybatis目录,然后创建SqlMapConfig.xml

将db.properties和log4j.properties拷贝到config目录下。
5.4.1.2Spring
在config下,创建spring目录,然后创建applicationContext.xml

5.4.2整合代码
5.4.2.1原始dao开发方式
5.4.2.1.1映射文件
在config/mybatis下创建sqlmap,然后创建User.xml

5.4.2.1.2Dao代码
Dao接口

Dao实现类

5.4.2.1.3配置UserDao实现类
在applicationContext.xml中配置UserDao实现类

5.4.2.1.4测试代码

5.4.2.2Mapper代理
5.4.2.2.1映射文件
将映射文件放到UserMapper接口的同包下

5.4.2.2.2Mapper接口

5.4.2.2.3配置mapper代理类
单个mapper代理类配置

批量设置mapper代理类

5.4.2.2.4测试代码

6逆向工程(会用)
6.1什么是逆向工程
Mybatis提供来一个逆向工程工具,通过逆向工程,可以帮助程序员根据单表来生成po类、mapper映射文件、mapper接口。

6.2下载逆向工程
https://github.com/mybatis/generator/releases/tag/mybatis-generator-1.3.2

6.3创建逆向工程

6.4创建Generator.java

6.5添加generatorConfig.xml

<?xml version="1.0" encoding="UTF-8"?>
	<!-- 默认false,把JDBC DECIMAL 和 NUMERIC 类型解析为 Integer,为 true时把JDBC DECIMAL 
		和 NUMERIC 类型解析为java.math.BigDecimal -->
	<javaTypeResolver>
		<property name="forceBigDecimals" value="false" />
	</javaTypeResolver>

	<!-- targetProject:生成PO类的位置 -->
	<javaModelGenerator targetPackage="com.itheima.ms.po"
		targetProject=".\src">
		<!-- enableSubPackages:是否让schema作为包的后缀 -->
		<property name="enableSubPackages" value="false" />
		<!-- 从数据库返回的值被清理前后的空格 -->
		<property name="trimStrings" value="true" />
	</javaModelGenerator>
	<!-- targetProject:mapper映射文件生成的位置 -->
	<sqlMapGenerator targetPackage="com.itheima.ms.mapper"
		targetProject=".\src">
		<!-- enableSubPackages:是否让schema作为包的后缀 -->
		<property name="enableSubPackages" value="false" />
	</sqlMapGenerator>
	<!-- targetPackage:mapper接口生成的位置 -->
	<javaClientGenerator type="XMLMAPPER"
		targetPackage="com.itheima.ms.mapper" targetProject=".\src">
		<!-- enableSubPackages:是否让schema作为包的后缀 -->
		<property name="enableSubPackages" value="false" />
	</javaClientGenerator>
	<!-- 指定数据库表 -->
	<table tableName="items"></table>
	<table tableName="orders"></table>
	<table tableName="orderdetail"></table>
	<table tableName="user"></table>
</context>

6.6将逆向工程生成的代码拷贝到指定项目中
6.7使用逆向工程生成的代码

6.8注意事项
Mapper.xml文件已经存在时,如果进行重新生成则mapper.xml文件时,内容不被覆盖而是进行内容追加,结果导致mybatis解析失败。
解决方法:删除原来已经生成的mapper xml文件再进行生成。
Mybatis自动生成的po及mapper.java文件不是内容而是直接覆盖没有此问题。

猜你喜欢

转载自blog.csdn.net/anhldd/article/details/84631674