动态sql的使用
where & if
<select id="findBlog" resultType="com.wei.pojo.Blog">
select * from blog
<where>
<if test="author!= null">
author = #{author}
</if>
</where>
</select>
select * from blog
- 如果传入的author是null,则会忽略,变成了查询所有
- 如果传入的不为空,则会查询对应的author,如果不存在就是空集合。
set & if
<update id="updateBlog" parameterType="map">
update blog
<set>
<if test="title != null">
title = #{title},
</if>
<if test="author != null">
author = #{author}
</if>
</set>
where id = #{id};
</update>
仅仅传递title
update blog SET title = ? where id = ?;
2个字段都不传递
update blog where id = ?;
- 动态生成update的sql,传递title和author的其中之一就可以修改,或者都传递。
- 如果二者都不传,则会出现错误。
choose & when & otherwise
<select id="queryBlogChoose" parameterType="map" resultType="blog">
select * from blog
<where>
<choose>
<when test="title != null">
title = #{title}
</when>
<when test="author != null">
and author = #{author}
</when>
<otherwise>
and views = #{views}
</otherwise>
</choose>
</where>
</select>
使用when标签里的属性的时候
select * from blog WHERE author = ? , title = ?
不使用when标签里的属性的时候使用,它会替我们去掉views 前的 and
select * from blog WHERE views = ?
- 可以选择title 或者 author 其中一个,或者全部进行查询 blog
- 如果两个都不选择,则会使用views来查询,在使用以上2个字段的情况下,不使用views
sql & include 标签 sql代码复用
<select id="findBlog" resultType="com.wei.pojo.Blog">
<include refid="selectBlog"/>
<where>
<if test="author!= null">
author = #{author}
</if>
</where>
</select>
<sql id="selectBlog">
select * from blog
</sql>
select * from blog
select * from blog where author = ?
sql标签使用的注意事项:
- 最好基于单表来定义SQL片段,有利于复用。
- 不要存在where标签
foreach
Java测试
// mapper
public interface BlogMapper {
List<Blog> selectBlogForEach(Map map);
}
//测试
@Test
public void selectBlogForEach(){
SqlSession sqlsession = MybatisUtils.getSqlsession();
BlogMapper mapper = sqlsession.getMapper(BlogMapper.class);
HashMap hashMap = new HashMap();
ArrayList<Integer> ids = new ArrayList();
ids.add(1);ids.add(2);
hashMap.put("ids",ids);
List<Blog> blogs = mapper.selectBlogForEach(hashMap);
}
<select id="selectBlogForEach" parameterType="map" resultType="com.wei.pojo.Blog">
select * from blog
<where>
<foreach collection="ids" open=" id in (" close=")" separator="," item="id">
#{id}
</foreach>
</where>
</select>
对应的sql语句
select * from blog WHERE id in ( ? , ? )
mybatis 缓存
- MyBatis包含一个非常强大的查询缓存特性,它可以非常方便地定制和配置缓存。缓存可以极大的提升查询效率。
- MyBatis系统中默认定义了两级缓存:一级缓存和二级缓存
- 默认情况下,只有一级缓存开启。(SqlSession级别的缓存,也称为本地缓存)
- 二级缓存需要手动开启和配置,他是基于namespace级别的缓存。
- 为了提高扩展性,MyBatis定义了缓存接口Cache。我们可以通过实现Cache接口来自定义二级缓存
- 经常改变的数据,不能用缓存。
一级缓存 sqlsession级别
缓存生效
@Test
public void fun1() {
SqlSession sqlSession = MybatisUtils.getSqlsession();
BlogMapper blogMapper = sqlSession.getMapper(BlogMapper.class);
Blog one = blogMapper.findOne(1);
Blog two = blogMapper.findOne(1);
System.out.println(one==two);//true,同一对象。
sqlSession.close();
}
缓存失效
1.两次查询之间出现增删改查语句
2.手动清除缓存、最近最少使用算法清除
3.sqlsession不同或sqlsession关闭后再次打开(同样是不同sqlsession)
4.不定时的刷新
@Test
public void fun2() {
SqlSession sqlSession = MybatisUtils.getSqlsession();
BlogMapper blogMapper = sqlSession.getMapper(BlogMapper.class);
Blog one = blogMapper.findOne(1);
sqlSession.close();//需要再次创建,但是结果还是false
sqlSession.clearCache();//清除了缓存,false
//增删改 //清除了缓存,false
System.out.println(one==two);//false
}
二级缓存 sqlsessionFactory级别
-
二级缓存也叫全局缓存,一级缓存作用域太低了,所以诞生了二级缓存
-
基于namespace级别的缓存,一个名称空间,对应一个二级缓存;
-
会话关闭了,对应的一级缓存就没了;会话关闭了,一级缓存中的数据被保存到二级缓存中;
-
新的会话查询信息,就可以从二级缓存中获取内容;不同的mapper查出的数据会放在自己对应的缓存(map)中;
如何开启二级缓存
1、开启全局缓存【mybatis-config.xml】
其实默认为true,可配可不配。
<setting name="cacheEnabled" value="true"/>
2、每个mapper.xml中配置使用二级缓存;
- 这个更高级的配置创建了一个 FIFO 缓存,每隔 60 秒刷新,最多可以存储结果对象或列表的 512 个引用,而且返回的对象被认为是只读的,因此对它们进行修改可能会在不同线程中的调用者产生冲突。
<cache eviction="FIFO" flushInterval="200000" readOnly="true" size="512"/>
<cache/> <!-- 2种 配置方式都可以。-->
3、代码测试
@Test
public void fun10() {
SqlSession sqlSession = MybatisUtils.getSqlsession();
SqlSession sqlSession2 = MybatisUtils.getSqlsession();
BlogMapper blogMapper = sqlSession.getMapper(BlogMapper.class);
BlogMapper blogMapper2 = sqlSession2.getMapper(BlogMapper.class);
Blog one = blogMapper.findOne(1);
sqlSession.close();//如果没有关闭sqlsession,则结果为false,因为二级缓存不存在
Blog two = blogMapper2.findOne(1);
System.out.println(one==two);//true,同一对象。
}
4、结论
- 只要开启了二级缓存,我们在同一个Mapper中的查询,可以在二级缓存中拿到数据
- 查出的数据都会被默认先放在一级缓存中
- 只有会话提交或者关闭以后,一级缓存中的数据才会转到二级缓存中
- 获取数据的顺序。先看二级缓存有没有缓存,再看一级缓存有没有缓存,再查询数据库