mybatis 知识点整理及补充

动态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
  1. 如果传入的author是null,则会忽略,变成了查询所有
  2. 如果传入的不为空,则会查询对应的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 = ?;
  1. 动态生成update的sql,传递title和author的其中之一就可以修改,或者都传递。
  2. 如果二者都不传,则会出现错误。

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 = ?
  1. 可以选择title 或者 author 其中一个,或者全部进行查询 blog
  2. 如果两个都不选择,则会使用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标签使用的注意事项:

  1. 最好基于单表来定义SQL片段,有利于复用。
  2. 不要存在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中的查询,可以在二级缓存中拿到数据
  • 查出的数据都会被默认先放在一级缓存中
  • 只有会话提交或者关闭以后,一级缓存中的数据才会转到二级缓存
  • 获取数据的顺序。先看二级缓存有没有缓存,再看一级缓存有没有缓存,再查询数据库
    在这里插入图片描述

猜你喜欢

转载自blog.csdn.net/qq_30081043/article/details/107372315