查询缓存
1.什么是查询缓存mybatis提供查询缓存,用于减轻数据压力,提高数据库性能。mybaits提供一级缓存,和二级缓存。缓存模式图如图
如图
//测试一级缓存@Testpublic void testCache1() throws Exception{ SqlSession sqlSession = sqlSessionFactory.openSession();//创建代理对象 UserMapper userMapper = sqlSession.getMapper(UserMapper.class); //下边查询使用一个SqlSession //第一次发起请求,查询id为1的用户 User user1 = userMapper.findUserById(1); System.out.println(user1.getUsername()); //第二次发起请求,查询id为1的用户 User user2 = userMapper.findUserById(1); System.out.println(user2.getUsername()); sqlSession.close(); }
测试结果和输出日志:
DEBUG [main] - Opening JDBC ConnectionDEBUG [main] - Created connection 19140890.DEBUG [main] - Setting autocommit to false on JDBC Connection [com.mysql.jdbc.Connection@124111a]DEBUG [main] - ==> Preparing: SELECT * FROM USER WHERE id=? DEBUG [main] - ==> Parameters: 1(Integer)DEBUG [main] - <== Total: 1张三张三DEBUG [main] - Resetting autocommit to true on JDBC Connection [com.mysql.jdbc.Connection@124111a]DEBUG [main] - Closing JDBC Connection [com.mysql.jdbc.Connection@124111a]DEBUG [main] - Returned connection 19140890 to pool.
可以看到第二次取user2的时候没有发出sql查询,说明调用了一级缓存
接下来测试看看在中间执行了增删改查的时候会不会清空一级缓存//测试一级缓存@Testpublic void testCache1() throws Exception{ SqlSession sqlSession = sqlSessionFactory.openSession();//创建代理对象 UserMapper userMapper = sqlSession.getMapper(UserMapper.class); //下边查询使用一个SqlSession //第一次发起请求,查询id为1的用户 User user1 = userMapper.findUserById(1); System.out.println(user1.getUsername()); /*如果sqlSession去执行commit操作(执行插入、更新、删除), * 清空SqlSession中的一级缓存,这样做的目的为了让缓存中存储的是最新的信息,避免脏读。 */ //更新user1的信息 user1.setUsername("测试用户22"); userMapper.updateUser(user1); //执行commit操作去清空缓存 sqlSession.commit(); //第二次发起请求,查询id为1的用户 User user2 = userMapper.findUserById(1); System.out.println(user2.getUsername()); sqlSession.close(); }
测试结果和输出日志:
DEBUG [main] - Opening JDBC ConnectionDEBUG [main] - Created connection 6778431.DEBUG [main] - Setting autocommit to false on JDBC Connection [com.mysql.jdbc.Connection@676e3f]DEBUG [main] - ==> Preparing: SELECT * FROM USER WHERE id=? DEBUG [main] - ==> Parameters: 1(Integer)DEBUG [main] - <== Total: 1张三DEBUG [main] - ==> Preparing: update user set username=?,birthday=?,sex=?,address=? where id=? DEBUG [main] - ==> Parameters: 测试用户2(String), 2015-06-07(Date), 男(String), 河南焦作(String), 1(Integer)DEBUG [main] - <== Updates: 1DEBUG [main] - Committing JDBC Connection [com.mysql.jdbc.Connection@676e3f]DEBUG [main] - ==> Preparing: SELECT * FROM USER WHERE id=? DEBUG [main] - ==> Parameters: 1(Integer)DEBUG [main] - <== Total: 1测试用户2DEBUG [main] - Resetting autocommit to true on JDBC Connection [com.mysql.jdbc.Connection@676e3f]DEBUG [main] - Closing JDBC Connection [com.mysql.jdbc.Connection@676e3f]DEBUG [main] - Returned connection 6778431 to pool.
发现第二次取数据时发出了sql请求,说明执行增删改查的时候会清空一级缓存
2.3一级缓存应用正式开发,是将mybatis和spring进行整合开发,事务控制在service中。一个service方法中包括 很多mapper方法调用。service{ //开始执行时,开启事务,创建SqlSession对象//第一次调用mapper的方法findUserById(1)//第二次调用mapper的方法findUserById(1),从一级缓存中取数据//方法结束,sqlSession关闭}如果是执行两次service调用查询相同 的用户信息,不走一级缓存,因为service方法结束,sqlSession就关闭,一级缓存就清空。这个时候就需要使用二级缓存,详情看下一篇总结