什么是查询缓存?
缓存是介于应用程序和物理数据源之间
mybatis提供查询缓存,用于减轻数据压力,提高数据库性能。
mybaits提供一级缓存,和二级缓存。
一级缓存是sqlSession级别的缓存。在操作数据库时需要构造sqlSession对象,在对象中有一个数据结构(HashMap),用于存储缓存数据。不同的sqlSession之间的缓存 区域(HashMap)是互不影响的。
二级缓存是mapper级别的缓存,多个sqlSession去操作同一个Mapper的sql语句,多个SqlSession可以公用二级缓存,二级缓存是跨sqlSession的
为什么要用缓存?
如果缓存中有数据就不用从数据库中获取,减少了和数据之间的交互次数,大大提高系统的性能
为什么会两种缓存方式?
二级缓存与一级缓存区别,二级缓存的范围更大,多个sqlSession可以共享一个UserMapper的二级缓存区域。
-
一级缓存
-
一级缓存工作原理
命中条件
缓存存在一个hash表中,通过查询SQL,查询数据库,客户端协议等作为key.在判断是否命中前,MySQL不会解析SQL,而是直接使用SQL去查询缓存,SQL任何字符上的不同,如空格,注释,都会导致缓存不命中.
如果查询中有不确定数据,例如CURRENT_DATE()和NOW()函数,那么查询完毕后则不会被缓存.所以,包含不确定数据的查询是肯定不会找到可用缓存的
-
1. 服务器接收SQL,以SQL和一些其他条件为key查找缓存表(额外性能消耗)
2. 如果找到了缓存,则直接返回缓存(性能提升)
3. 如果没有找到缓存,则执行SQL查询,包括原来的SQL解析,优化等.
4. 执行完SQL查询结果以后,将SQL查询结果存入缓存表(额外性能消耗)
为什么sqlSession去执行commit操作(执行插入、更新、删除),清空SqlSession中的一级缓存,目的为了让缓存中存储的是最新的信息,
避免脏读。(所谓的脏数据)
-
-
一级缓存测试
-
mybatis默认支持一级缓存,不需要在配置文件去配置。
测试代码
1 // 一级缓存测试 2 @Test 3 public void testCache1() throws Exception { 4 SqlSession sqlSession = sqlSessionFactory.openSession();// 创建代理对象 5 UserMapper userMapper = sqlSession.getMapper(UserMapper.class); 6 7 // 下边查询使用一个SqlSession 8 // 第一次发起请求,查询id为1的用户 9 User user1 = userMapper.getUserById(1); 10 System.out.println(user1); 11 12 // 如果sqlSession去执行commit操作(执行插入、更新、删除),清空SqlSession中的一级缓存,这样做的目的为了让缓存中存储的是最新的信息,避免脏读。 13 14 // 更新user1的信息 15 // user1.setUsername("我是测试用户"); 16 // userMapper.updateUser(user1); 17 // //执行commit操作去清空缓存 18 // sqlSession.commit(); 19 20 // 第二次发起请求,查询id为1的用户 21 User user2 = userMapper.getUserById(1); 22 System.out.println(user2); 23 24 sqlSession.close(); 25 26 }
假设我们不执行更新操作,输出:
1 DEBUG [main] - Opening JDBC Connection 2 DEBUG [main] - Created connection 110771485. 3 DEBUG [main] - Setting autocommit to false on JDBC Connection [com.mysql.jdbc.JDBC4Connection@69a3d1d] 4 DEBUG [main] - ==> Preparing: SELECT * FROM user WHERE id=? 5 DEBUG [main] - ==> Parameters: 1(Integer) 6 DEBUG [main] - <== Total: 1 7 User [id=1, username=张三, sex=男] 8 User [id=1, username=张三, sex=男] 9 DEBUG [main] - Resetting autocommit to true on JDBC Connection [com.mysql.jdbc.JDBC4Connection@69a3d1d] 10 DEBUG [main] - Closing JDBC Connection [com.mysql.jdbc.JDBC4Connection@69a3d1d] 11 DEBUG [main] - Returned connection 110771485 to pool.
使用上面更新的代码,输出:
1 DEBUG [main] - Opening JDBC Connection 2 DEBUG [main] - Created connection 110771485. 3 DEBUG [main] - Setting autocommit to false on JDBC Connection [com.mysql.jdbc.JDBC4Connection@69a3d1d] 4 DEBUG [main] - ==> Preparing: SELECT * FROM user WHERE id=? 5 DEBUG [main] - ==> Parameters: 1(Integer) 6 DEBUG [main] - <== Total: 1 7 User [id=1, username=张三, sex=男] 8 DEBUG [main] - ==> Preparing: update user set username=?,sex=? where id=? 9 DEBUG [main] - ==> Parameters: 我是测试用户(String), null, 1(Integer) 10 DEBUG [main] - <== Updates: 1 11 DEBUG [main] - Committing JDBC Connection [com.mysql.jdbc.JDBC4Connection@69a3d1d] 12 DEBUG [main] - ==> Preparing: SELECT * FROM user WHERE id=? 13 DEBUG [main] - ==> Parameters: 1(Integer) 14 DEBUG [main] - <== Total: 1 15 User [id=1, username= 我是测试用户, sex=男] 16 DEBUG [main] - Resetting autocommit to true on JDBC Connection [com.mysql.jdbc.JDBC4Connection@69a3d1d] 17 DEBUG [main] - Closing JDBC Connection [com.mysql.jdbc.JDBC4Connection@69a3d1d] 18 DEBUG [main] - Returned connection 110771485 to pool.
原来用户名为张三则更新为我是测试用户
-
-
一级缓存应用
-
正式开发,假如项目是将mybatis和spring进行整合开发....,事务控制在service中。
一个service方法中包括 很多mapper方法调用。
1 service{ 2 3 //开始执行时,开启事务,创建SqlSession对象 4 5 //第一次调用mapper的方法getUserById(1) 6 7 //第二次调用mapper的方法getUserById(1),从一级缓存中取数据 8 9 //方法结束,sqlSession关闭 10 11 }
如果是执行两次service调用查询相同 的用户信息,不走一级缓存,因为session方法结束,sqlSession就关闭,一级缓存就清空。
-
-
缓存数据失效时机
-
在表的结构或数据发生改变时,查询缓存中的数据不再有效。有这些INSERT、UPDATE、 DELETE、TRUNCATE、ALTER TABLE、
DROP TABLE或DROP DATABASE会导致缓存数据失效。所以查询缓存适合有大量相同查询的应用,不适合有大量数据更新的应用。
-
- 可以使用下面三个SQL来清理查询缓存:
-
二级缓存
- 二级缓存的工作原理
第一步:首先开启mybatis的二级缓存。
第二步:sqlSession1去查询用户id为1的用户信息,查询到用户信息会将查询数据存储到二级缓存中。
第三步:如果SqlSession3去执行相同 mapper下sql,执行commit提交,清空该 mapper下的二级缓存区域的数据。
第四步:sqlSession2去查询用户id为1的用户信息,去缓存中找是否存在数据,如果存在直接从缓存中取出数据。
UserMapper有一个二级缓存区域(按namespace分) ,其它mapper也有自己的二级缓存区域(按namespace分)。
每一个namespace的mapper都有一个二缓存区域,两个mapper的namespace如果相同,这两个mapper执行sql查询
到数据将存在相同的二级缓存区域中。
- 开启二级缓存
mybaits的二级缓存是mapper范围级别,除了在SqlMapConfig.xml设置二级缓存的总开关,还要在具体的mapper.xml中开启二级缓存。
第一步:在核心配置文件mybatis-config.xml中加入以下代码
1 <!-- 全局参数的配置 --> 2 <settings> 3 <!-- 开启二级缓存 --> 4 <setting name="cacheEnabled" value="true"/> 5 </settings>
cacheEnabled:对在此配置文件下的所有cache 进行全局性开/关设置。默认true
第二步:实体类User实现序列化接口
1 public class User implements Serializable{ 2 ...... 3 4 }
目的:为了将缓存数据取出执行反序列化操作,因为二级缓存数据存储介质多种多样,不一样在内存。
第三步:使用Junit进行测试:
1 // 二级缓存测试 2 @Test 3 public void testCache2() throws Exception { 4 SqlSession sqlSession1 = sqlSessionFactory.openSession(); 5 SqlSession sqlSession2 = sqlSessionFactory.openSession(); 6 SqlSession sqlSession3 = sqlSessionFactory.openSession(); 7 8 // 创建代理对象 9 UserMapper userMapper1 = sqlSession1.getMapper(UserMapper.class); 10 // 第一次发起请求,查询id为1的用户 11 User user1 = userMapper1.getUserById(1); 12 System.out.println(user1); 13 14 //这里执行关闭操作,将sqlsession中的数据写到二级缓存区域 15 sqlSession1.close(); 16 17 18 //使用sqlSession3执行commit()操作 19 UserMapper userMapper3 = sqlSession3.getMapper(UserMapper.class); 20 User user = userMapper3.getUserById(1); 21 user.setUsername("张明明");
userMapper3.updateUser(user); 22 //执行提交,清空UserMapper下边的二级缓存 23 sqlSession3.commit(); 24 sqlSession3.close(); 25 26 UserMapper userMapper2 = sqlSession2.getMapper(UserMapper.class); 27 // 第二次发起请求,查询id为1的用户 28 User user2 = userMapper2.getUserById(1); 29 System.out.println(user2); 30 31 sqlSession2.close(); 32 33 }
- 禁用二级缓存
在statement中设置useCache=false可以禁用当前select语句的二级缓存,即每次查询都会发出sql去查询,
默认情况是true,即该sql使用二级缓存。
<select id="findOrderListResultMap" resultMap="ordersUserMap" useCache="false">
- 刷新缓存
在mapper的同一个namespace中,如果有其它insert、update、delete操作数据后需要刷新缓存,如果不执行刷新缓存会出现脏读。
设置statement配置中的flushCache="true" 属性,默认情况下为true即刷新缓存,如果改成false则不会刷新。使用缓存时如果手动
修改数据库表中的查询数据会出现脏读。
如下:
<insert id="insertUser" parameterType="com.mybaits.entity.User" flushCache="true">
- 二级缓存应用场景
对于访问多的查询请求且用户对查询结果实时性要求不高,此时可采用mybatis二级缓存技术降低数据库访问量,
提高访问速度,业务场景比如:耗时较高的统计分析sql、电话账单查询sql等。
实现方法如下:通过设置刷新间隔时间,由mybatis每隔一段时间自动清空缓存,根据数据变化频率设置缓存刷新间
隔 flushInterval ,比如设置为30分钟、60分钟、24小时等,根据需求而定。
- 二级缓存的局限性
mybatis二级缓存对细粒度的数据级别的缓存实现不好,比如如下需求:对商品信息进行缓存,由于商品信息查询访问量大,
但是要求用户每次都能查询最新的商品信息,此时如果使用mybatis的二级缓存就无法实现当一个商品变化时只刷新该商品的缓存
信息而不刷新其它商品的信息,因为mybaits的二级缓存区域以mapper为单位划分,当一个商品信息变化会将所有商品信息的缓存数据
全部清空。解决此类问题需要在业务层根据需求对数据有针对性缓存
vue是什么?
vue是一个当前很火的js框架。它可以将我们的数据,和显示数据的DOM文档进行绑定。一旦绑定以后,DOM和数据将会自动同步。使我们不用在考虑给DOM的某个元素去进行赋值,而是将注意力放在数据模型上。