MyBatis —— 复习

这篇博客详细复习了MyBatis的使用步骤,包括配置解析、SqlSessionFactory和SqlSession的生命周期及作用域。重点介绍了ResultMap的映射机制、日志配置和Log4j的使用。还涵盖了MyBatis的执行流程、@Param注解、#{}与${}的区别、Lombok的注解以及动态SQL。同时,文章深入讲解了MyBatis的缓存机制,包括一级缓存和二级缓存的工作原理、开启与配置、缓存失效情况和清除策略。

摘要生成于 C知道 ,由 DeepSeek-R1 满血版支持, 前往体验 >

MyBatis复习

1.代码中使用mybatis的步骤

//1.获取SqlSessionFactory对象
InputStream inputStream = Resource.getResourceAsStream("mybatis-config");
SqlSessionFactory sqlSessionFactory = new SqlSessionFactoryBuilder().build(inputStream);
//2.通过SqlSessionFactory获取SqlSession
SqlSession sqlSession = sqlSessionFactory.openSession();
//3.getMapper
XxxMapper xxxMapper = sqlSession.getMapper(XxxMpper.class);
//4.调用Mapper中的方法即可
List<Xxx> xxxList = xxxMapper.getList();

2.配置解析

configuration(配置)
properties(属性)
settings(设置)
typeAliases(类型别名)
typeHandlers(类型处理器)
objectFactory(对象工厂)
plugins(插件)
environments(环境配置)
environment(环境变量)
transactionManager(事务管理器)
dataSource(数据源)
databaseIdProvider(数据库厂商标识)
mappers(映射器)

3.生命周期和作用域

生命周期和作用于是十分重要的,错误的使用会导致并发问题
SqlSessionFactoryBuilder

  • 一旦创建SqlSessionFactory,就不在需要它了
  • 局部变量

SqlSessionFactory

  • 可以理解为数据库连接池
  • 一旦创建就应该在应用的运行期间一直存在,没有任何理由丢弃它或重新创建另一个实例
  • 最佳作用域是一种应用作用域
  • 最简单的就是使用单例模式或静态单例模式

SqlSession

  • 可以理解为连接到连接池的一个请求
  • 需要开启/关闭请求
  • 最佳作用域是放在方法中,用完之后赶紧关闭,否则资源占用
  • 不是线程安全的,因此是不能共享的

4.结果集映射ResultMap

设计思想:对于简单的语句根本不需要配置显示的结果映射,而对于复杂一点你的语句只需要描述它们的关系即可


5.日志配置

在mybatis-config中可以进行日志的配置

<settings>
	<setting name="logImpl" value="STDOUT_LOGGING"/>
</settings>

6.Log4j

  • Log4j是Apache的一个开源项目,通过使用Log4j,我们可以控制日志信息输送的目的是控制台、文件、GUI组件
  • 可以控制每一条日志的输出格式
  • 通过定义每一条日志信息的级别,我们能够更加细致地控制日志的生成过程
  • 通过一个配置文件来灵活的进行配置,而不需要修改应用的代码

具体使用:

  • 先导入log4j的依赖
<dependency>
      <groupId>log4j</groupId>
      <artifactId>log4j</artifactId>
      <version>1.2.17</version>
</dependency>
  • 配置log4j.properties
  • 在mybatis配置文件中配置log4j为日志的实现
  • 使用即可

日志级别:

  • info
  • debug
  • error

7.Mybatis详细执行流程

  • Resource获取加载全局配置文件
  • 实例化SqlSessionFactoryBuilder构造器
  • 解析配置文件流 XML ConfigBuilder
  • Configuration所有的配置信息
  • SqlSessionFactory实例化
  • transactional事务管理器
  • 创建executor执行器
  • 创建SqlSession
  • 实现CRUD(出现问题会回滚,回到到事务管理器)
  • 查看是否成功(出现问题会回滚,回到到事务管理器)
  • 提交事务
  • 关闭

8.@Param()注解

  • 基本类型的参数或者String类型,需要加上
  • 引用类型不需要加
  • 如果只有一个基本类型的话,可以忽略,但是建议加上
  • 在SQL中引用的就是@Param()中设定的属性名

9.#{}和${}

  • #{}可以很大程度防止SQL注入问题;${}不能防止SQL注入问题
  • ${}是进行字符串拼接

10.Lombok

  • @Data
  • @AllargsConstructor
  • @NoArgsConstructor
  • @EqualsAndHashCode
  • @ToString
  • @Getter

11.多对一和多对多

在sql.xml中,涉及到多对多、多对一关系,需要在ResultMap中手动映射关系
单个属性使用result、对象使用association、集合使用collection
多对多:
嵌套子查询使用时:
association:javaType为对象的类型;select为嵌套查询语句的id
连表查询查询时:
association:javaType为对象的类型;

<association>
	<result/>
</association>

一对多:
连表查询查询时:

<collection>
	<result/>
</collection>

嵌套子查询使用时:
collection:javaType为ArrayList;oftype为集合中泛型的类型;select为嵌套查询语句的id


12.动态sql

<!-- if -->
<if test="表达式返回真假">
	SQL语句
</if>

<!-- where -->
<where>
	SQL语句或其他的判断逻辑,语句开头若有and或者or,where标签会自动去除
</where>

<!-- choose when -->
<where>
	<choose> 
		<when test="表达式"> </when>
		<otherwise> </otherwise>
	</choose>
</where>

<!-- set -->
update 表名
	<set>
		修改的字段
	</set>
	
<!-- sql -->
<sql id="aaa">
</sql>
<include refid="aaa"/>

<!-- foreach -->
<foreach item="item" index="index" collection="list" open="(" separator="," close=")">
	#{item}
</foreach

13.Mybatis缓存

什么是缓存?

  • 存在内存中的临时数据
  • 将用户经常查询的数据放在缓存中,用户去查询数据就不用从磁盘上查询,从缓存中查询,从而提高查询效率,解决了高并发系统的性能问题

为什么使用缓存? 减少和数据库的交互次数,减少系统开销,提高系统效率
什么样的数据能使用缓存? 经常查询并且不经常改变的数据

Mybtais包含了一个非常强大的查询缓存特性,它可以非常方便地定制和配置缓存,缓存可以极大的提升查询效率
Mybatis系统中默认定义了两级缓存:一级缓存和二级缓存

  • 默认情况下,只有一级缓存开启(SqlSession级别的缓存,也称为本地缓存)
  • 二级缓存需要手动开启和配置,他是基于namespace级别的缓存
  • 为了提高扩展性,Mybatis定义了缓存接口Cache,可以通过Cache接口来自定义二级缓存

一级缓存:
一级缓存也叫本地缓存

  • 与数据库同一次会话期间查询到的数据会放在本地缓存中
  • 以后如果需要获取相同的数据,直接从缓存中拿,没必要再去缓存数据库

测试一级缓存:

  • 开启日志
  • 测试在一个session中查询两次相同的记录
  • 查看日志输出可以看出SQL只执行了一次,并且两次返回的对象相等

缓存失效的情况:

  • 增删改操作,可能会改变原来操作,所以必定会刷新缓存
  • 查询不同的东西
  • 查询不同的mapper.xml
  • 手动清理缓存sqlSession.clearCache();

一级缓存默认是开启的妹纸在一次SqlSession中有效,也就是拿到连接到关闭连接这个区间段
二级缓存:

  • 二级缓存也叫全局缓存,一级缓存作用域太低了,所以诞生了二级缓存
  • 基于namespace级别的缓存,一个名称空间,对应一个二级缓存
  • 工作机制:
    • 一个会话查询一条数据,这个数据就会被放在当前会话的一级缓存中
    • 如果当前会话关闭了,这个会话对应的一级缓存就没了;但是我们想要的是,会话关闭了,一级缓存中的数据被保存到二级缓存中
    • 新的会话查询信息,就可以从二级缓存中获取内容
    • 不同的mapper查出的数据会放在自己对应的缓存中

二级缓存使用:

  • 开启全局缓存
<!-- 在mybatis配置文件中开启全局缓存 -->
<setting name="cacheEnabled" value="true"/>
  • 需要再mapper.xml中添加一个标签<cache/>
    标签中可以添加一些属性(不添加属性也可以使用):
`<cache 
	eciction="FIFO"  FIFO缓存
	flushInterval="60000"   每隔60秒刷新
	size="512"   最多可以存储结果对象或列表512个
	readOnly="true"/>	返回对象被认为是只读,因此对它们进行修改可能会在不同线程中的调用者产生冲突

注意: 使用cache,一定要将实体序列化

二级缓存是事务性的,这意味着,当SqlSession完成并提交时,或是完成并回滚,但没有执行flushCache=true的insert/dalete/update语句时,缓存会获得更新

清除策略:

  • LUR 最近最少使用:移除最长时间不被使用的对象 (cache默认清除策略)
  • FIFO 先进先出:按对象进入缓存的顺序来移除它们
  • SOFT 软引用:基于垃圾回收器状态和软引用规则移除对象
  • WEAK 弱引用:更积极地基于垃圾收集器状态和弱引用规则移除对象

缓存原理:

  • 先去二级缓存中查询
  • 二级缓存没有再去一级缓存
  • 到一级缓存和二级缓存中都没查到,再去数据库中查
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值