Mybatis中的缓存
引言
什么是缓存:
     存在于内存中的临时数据。
为什么使用缓存:
    减少和数据库的交互次数,提高执行效率。
什么样的数据能用缓存,什么数据不能使用:
    适用于缓存:
        经常查询并且不经常改变的
        数据的正确与否对最终结果影响不大(因为缓存代表着会与数据库数据不同)
不适用于缓存:
经常改变的数据
数据的正确与否对最终结果影响很大的
例如:商品的库存,银行的汇率,股市的牌价
一级缓存
定义
mybatis中SqlSession对象的缓存。当我们执行查询之后,查询的结果会同时存入到SqlSession为我们提供的一块区域中。该区域的结构是一个Map。当我们再次查询同样的数据,mybatis会先去SqlSession中查看是否有,有的话直接拿出来用。
当SqlSession对象消失时,一级缓存也就消失了。
一级缓存演示
    @Test
    public void testFirstLevelCache(){
        User user1 = dao.findById(41);
        System.out.println(user1);
        User user2 = dao.findById(41);
        System.out.println(user2);
        System.out.println(user1 == user2);
    }
只进行一次查询,第二次查询使用的是一级缓存,获取的对象也是相同地址的。

    @Test
    public void testFirstLevelCache(){
        User user1 = dao.findById(41);
        System.out.println(user1);
        session.close();
        session = factory.openSession();
        dao = session.getMapper(UserDao.class);
        /**可用:session.clearCache(); 清空缓存**/
        User user2 = dao.findById(41);
        System.out.println(user2);
        System.out.println(user1 == user2);
    }
执行了两次查询,对象地址不相同,证明第二个user是再new一次的


如果一级缓存中的数据域数据库不一致如何做到同步
更新办法
    <update id="updateUser" parameterType="user">
        update user set id = #{id}
        <if test="username != null and username != ''">
            ,username = #{username}
        </if>
        <if test="address != null and address != ''">
            ,address = #{address}
        </if>
        <if test="sex != null and sex != ''">
            ,sex = #{sex}
        </if>
        <if test="birthday != null">
            ,birthday = #{birthday}
        </if>
         where id = #{id}
    </update>
演示
    @Test
    public void testFirstLevelCache(){
        User user1 = dao.findById(41);
        System.out.println(user1);
        User user = new User();
        user.setId(41);
        user.setUsername("clear cache");
        user.setSex("女");
        dao.updateUser(user);
        User user2 = dao.findById(41);
        System.out.println(user2);
        System.out.println(user1 == user2);
    }
可以看出更新后,就不会从一级缓存中查找,而是进行一次新的查询

原因
当调用SqlSession的修改,添加,删除,commit(),close()等方法时,就会清空一级缓存
二级缓存
定义
指的是Mybatis中SqlSessionFactory对象的缓存。由同一个SqlSessionFactory对象创建的SqlSession共享其缓存。

二级缓存的使用步骤
第一步:让mybatis框架支持二级缓存(在SqlMapConfig.xml中配置)
    <settings>
        <setting name="cacheEnabled" value="true"/>
    </settings>
第二步:让当前的映射文件支持二级缓存(在UserDao.xml中配置)
    <!--开启user支持二级缓存-->
    <cache></cache>
@CacheNamespace(blocking = true)
public interface UserDao {
  ...
  ...
  ...
}
第三步:让当前的操作支持二级缓存(在select标签中配置)(注解配置不需要此步)
    <!--根据id查询单个用户-->
    <select id="findById" resultType="com.czy.domain.User" parameterType="INT" useCache="true">
        select * from user where id = #{id};
    </select>
测试
    @Test
    public void testSecondLevelCache(){
        SqlSession session1 = factory.openSession();
        UserDao dao1 = session1.getMapper(UserDao.class);
        User user1 = dao1.findById(41);
        System.out.println(user1);
        session1.close(); //一级缓存消失
        SqlSession session2 = factory.openSession();
        UserDao dao2 = session2.getMapper(UserDao.class);
        User user2 = dao2.findById(41);
        System.out.println(user2);
        session2.close();
        System.out.println(user1 == user2);
    }
只进行了一次查询,第二次直接调用缓存,由于存储的是数据不是对象,故比较结果为false


 
                
            
         
         浙公网安备 33010602011771号
浙公网安备 33010602011771号