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 弱引用:更积极地基于垃圾收集器状态和弱引用规则移除对象
缓存原理:
- 先去二级缓存中查询
- 二级缓存没有再去一级缓存
- 到一级缓存和二级缓存中都没查到,再去数据库中查