牢记MyBatis的一级、二级缓存和序列化问题
创始人
2025-05-30 01:14:06

缓存就是内存中的一个空间,通常用来提高查询效率。MyBatis支持两种缓存技术:一级缓存和二级缓存。其中一级缓存默认开启,二级缓存默认关闭。

一级缓存:

  1. 一级缓存默认开启

  2. 是一种基于SqlSession的缓存,同一个SqlSession有效,不同的SqlSession无效

  3. 同一个id,可以走缓存;不同id,即使查询的内容相同,也不会走缓存示例代码如下:

  • 示例代码如下:

    //加载核心配置文件
    InputStream is = Resources.getResourceAsStream("mybatis-cfg.xml");
    //基于构建者创建工厂对象
    SqlSessionFactory factory = new SqlSessionFactoryBuilder().build(is);
    //通过工厂创建sqlSession对象:测试一级缓存
    //第一次查询
    SqlSession sqlSession1 = factory.openSession();
    List list1 = sqlSession1.selectList("selAll1");
    System.out.println(list1);
    System.out.println("*************************");
    //第二次查询
    List list2 = sqlSession1.selectList("selAll1");
    System.out.println(list2);
    sqlSession1.close();
    

       mybatis一级缓存
  1. //加载核心配置文件
    InputStream is = Resources.getResourceAsStream("mybatis-cfg.xml");
    //基于构建者创建工厂对象
    SqlSessionFactory factory = new SqlSessionFactoryBuilder().build(is);
    //通过工厂创建sqlSession对象:测试一级缓存
    SqlSession sqlSession1 = factory.openSession();
    List list1 = sqlSession1.selectList("selAll1");
    System.out.println(list1);
    System.out.println("***********不同id,即使内容相同,也不走缓存**************");
    List list2 = sqlSession1.selectList("selAll2");
    System.out.println(list2);
    sqlSession1.close();
    

  2. mybatis一级缓存
    //加载核心配置文件
    InputStream is = Resources.getResourceAsStream("mybatis-cfg.xml");
    //基于构建者创建工厂对象
    SqlSessionFactory factory = new SqlSessionFactoryBuilder().build(is);
    //通过工厂创建sqlSession对象:测试一级缓存
    SqlSession sqlSession1 = factory.openSession();
    List list1 = sqlSession1.selectList("selAll1");
    System.out.println(list1);
    //        sqlSession1.close();
    System.out.println("***********不是同一个sqlSession,不会走缓存**************");
    SqlSession sqlSession2 = factory.openSession();
    List list2 = sqlSession2.selectList("selAll1");
    //        System.out.println(list2);
    

    mybatis一级缓存
  • 二级缓存:

  1. 二级缓存默认关闭

  2. 是一种基于SqlSessionFactory的缓存,同一个工厂创建的SqlSession有效,不同工厂则无效

  3. 需要使用时,应在指定的namespace下进行开启

  4. 标签的属性

  • eviction:缓存清除策略,常用的有LRU、FIFO、SOFT、WEAK

  • flushInterval:刷新间隔,需要给定一个毫秒数,表示指定时间间隔时,会自动刷新缓存

  • size:设置大小,默认是1024

  • readOnly:是否只读(不写入文件),默认是false,需要实体类实现序列化接口

  • type:用于自定义缓存机制,提供一个自定义类的全限定路径,自定义类需要实现CaChe接口

  • type:用于自定义缓存机制,提供一个自定义类的全限定路径,自定义类需要实现CaChe接口

//加载核心配置文件
InputStream is = Resources.getResourceAsStream("mybatis-cfg.xml");
//基于构建者创建工厂对象
SqlSessionFactory factory = new SqlSessionFactoryBuilder().build(is);
//通过工厂创建sqlSession对象:测试二级缓存
SqlSession sqlSession1 = factory.openSession();List list1 = sqlSession1.selectList("selAll1");
System.out.println(list1);
sqlSession1.close();System.out.println("***********同一个工厂的sqlSession,会走缓存**************");
SqlSession sqlSession2 = factory.openSession();
List list2 = sqlSession2.selectList("selAll1");
System.out.println(list2);

二级缓存

cache hit ratio中文意思是缓存命中率

这是一个计算机术语,终端用户访问加速节点时,如果该节点有缓存住了要被访问的数据时就叫做命中,如果没有的话需要回原服务器取,就是没有命中。

序列化问题:

//加载核心配置文件
InputStream is = Resources.getResourceAsStream("mybatis-cfg.xml");
//基于构建者创建工厂对象
SqlSessionFactory factory = new SqlSessionFactoryBuilder().build(is);
//通过工厂创建sqlSession对象:测试二级缓存
SqlSession sqlSession1 = factory.openSession();List list1 = sqlSession1.selectList("selAll1");
System.out.println(list1);
sqlSession1.close();

序列化问题

原因分析:User实现类未实现序列化。

为什么要实现序列化:(1)缓存机制:将查询结果保存到内存中 (2)内存饱满,需要移出时,MyBatis就会自动将内存中的内容进行移除,但是文件很重要,不能删除,此时就需要进行序列化,以文件的形式将内容从内存保存到硬盘上,一个内容保存成文件的读写,必须实现序列化。

解决方案:

一:User实体类实现序列化

序列化

二:在标签中添加readOnly属性

表示:要求Mybatis对缓存内容只读不写,当需要移除的时候,直接删除,不需要进行转存。

相关内容

热门资讯

桐昆股份股价涨5.02%,国投... 12月17日,桐昆股份涨5.02%,截至发稿,报15.05元/股,成交3.58亿元,换手率1.02%...
海光信息股价涨5.04%,华泰... 12月17日,海光信息涨5.04%,截至发稿,报207.71元/股,成交32.87亿元,换手率0.7...
广发证券股价涨5.05%,农银... 12月17日,广发证券涨5.05%,截至发稿,报22.25元/股,成交15.50亿元,换手率1.21...
胜宏科技股价涨5.04%,嘉合... 12月17日,胜宏科技涨5.04%,截至发稿,报301.55元/股,成交101.97亿元,换手率4....
胜宏科技股价涨5.04%,宝盈... 12月17日,胜宏科技涨5.04%,截至发稿,报301.55元/股,成交102.02亿元,换手率4....