Mybatis 一、二级缓存源码解析_mybatis二级缓存源码-程序员宅基地

技术标签: java  缓存  开发语言  


一. 什么是Mybatis缓存

缓存就是内存中的数据,常常来自对数据库查询结果的保存,使用缓存,我们可以避免频繁的与数据库进行交互,进而提高响应速度。

mybatis也提供了对缓存的支持,分为一级缓存和二级缓存,可以通过下图来理解:
在这里插入图片描述
①、一级缓存是SqlSession级别的缓存。在操作数据库时需要构造sqlSession对象,在对象中有一个数据结构(HashMap)用于存储缓存数据。不同的sqlSession之间的缓存数据区域(HashMap)是互相不影响的。
②、二级缓存是mapper级别的缓存,多个SqlSession去操作同一个Mapper的sql语句,多个SqlISession可以共用二级缓存,二级缓存是跨SqlSession的


二. 一级缓存

2.1 一级缓存演示

在这里插入图片描述

  1. 第⼀次发起查询⽤户id为1的⽤户信息,先去找缓存中是否有id为1的⽤户信息,如果没有,从数据
    库查询⽤户信息。得到⽤户信息,将⽤户信息存储到⼀级缓存中。
  2. 如果中间sqlSession去执⾏commit操作(执⾏插⼊、更新、删除),则会清空SqlSession中的⼀
    级缓存,这样做的⽬的为了让缓存中存储的是最新的信息,避免脏读。
  3. 第⼆次发起查询⽤户id为1的⽤户信息,先去找缓存中是否有id为1的⽤户信息,缓存中有,直接从
    缓存中获取⽤户信息

2.2 一级缓存是什么?

一提到⼀级缓存就绕不开SqlSession,所以索性我们就直接从SqlSession这个接口入手,看看有没有创建缓存或者与缓存有关的属性或者⽅法:
在这里插入图片描述
通过查找sqlsession接口的抽象方法,我们发现只有一个clearCache()的方法和缓存有点关系,那么我们可以点击这个方法的实现类查看,

在这里插入图片描述

通过上图的流程分析我们可以看出:Mybatis一级缓存的底层实现其实是一个Object类型的HashMap。

每一个sqlsession都会存在一个map对象的引用

分析了一圈我们可以的得到以下的流程图:
在这里插入图片描述

2.3 一级缓存什么时候被创建的?

我们推测一级缓存的创建,肯定是在刚刚罗列的几个类中的其中一个执行
在这里插入图片描述

通过我们的分析,我们在Executor.class这个类中发现一个createCacheKey的方法,从字面意思上看 很像是一级缓存创建的方法

在这里插入图片描述

我们点击createCacheKey的实现类BaseExecutor,查看该接口的具体实现

 
	//RowBounds: 分页对象
	//BoundSql :待执行的SQL语句存在这个对象中
    public CacheKey createCacheKey(MappedStatement ms, Object parameterObject, RowBounds rowBounds, BoundSql boundSql) {
    
        if (this.closed) {
    
            throw new ExecutorException("Executor was closed.");
        } else {
    
        	//创建cacheKey对象
            CacheKey cacheKey = new CacheKey();
            //ms.getId(): namespace.id
            cacheKey.update(ms.getId());
            //设置分页参数
            cacheKey.update(rowBounds.getOffset());
            cacheKey.update(rowBounds.getLimit());
            //获取到待执行的sql
            cacheKey.update(boundSql.getSql());

			//综上所述:cacheKey = namespace.id + 分页参数 +待执行的sql语句

            List<ParameterMapping> parameterMappings = boundSql.getParameterMappings();
            TypeHandlerRegistry typeHandlerRegistry = ms.getConfiguration().getTypeHandlerRegistry();
            Iterator var8 = parameterMappings.iterator();
			//设置参数的 忽略
            while(var8.hasNext()) {
    
                ParameterMapping parameterMapping = (ParameterMapping)var8.next();
                if (parameterMapping.getMode() != ParameterMode.OUT) {
    
                    String propertyName = parameterMapping.getProperty();
                    Object value;
                    if (boundSql.hasAdditionalParameter(propertyName)) {
    
                        value = boundSql.getAdditionalParameter(propertyName);
                    } else if (parameterObject == null) {
    
                        value = null;
                    } else if (typeHandlerRegistry.hasTypeHandler(parameterObject.getClass())) {
    
                        value = parameterObject;
                    } else {
    
                        MetaObject metaObject = this.configuration.newMetaObject(parameterObject);
                        value = metaObject.getValue(propertyName);
                    }

                    cacheKey.update(value);
                }
            }
            //判断配置文件中是否存在数据库连接的四大参数
            if (this.configuration.getEnvironment() != null) {
    
                cacheKey.update(this.configuration.getEnvironment().getId());
            }

            return cacheKey;
        }
    }

点进去 cacheKey.update(this.configuration.getEnvironment().getId())的update方法,在这个方法中,完成了cacheKey的封装

在这里插入图片描述

那么接下来,又有一个疑问了 Executor.class这个类中的createCacheKey方法是什么时候被调用的?按照正常的逻辑来说,是不是应该先从数据库中查询到数据,然后再将这个数据保存到缓存中呢?所以我们要了解这个createCacheKey方法是什么时候被调用的,应该要去找到底层执行sql的方法

我们知道只要是执行查询的操作,底层执行的都是Executor.class中的query方法 —> 查看该方法的实现


	//BaseExecutor中query方法的具体实现
    public <E> List<E> query(MappedStatement ms, Object parameter, RowBounds rowBounds, ResultHandler resultHandler) throws SQLException {
    
    	//获取到待指向的sql语句
        BoundSql boundSql = ms.getBoundSql(parameter);
        //开始调用createCacheKey方法
        CacheKey key = this.createCacheKey(ms, parameter, rowBounds, boundSql);
        //点击this.query(ms, parameter, rowBounds, resultHandler, key, boundSql)方法继续查看
        return this.query(ms, parameter, rowBounds, resultHandler, key, boundSql);
    }
    

this.query(ms, parameter, rowBounds, resultHandler, key, boundSql)方法的源码,如下:


 public <E> List<E> query(MappedStatement ms, Object parameter, RowBounds rowBounds, ResultHandler resultHandler, CacheKey key, BoundSql boundSql) throws SQLException {
    
        ErrorContext.instance().resource(ms.getResource()).activity("executing a query").object(ms.getId());
        if (this.closed) {
    
            throw new ExecutorException("Executor was closed.");
        } else {
    
            if (this.queryStack == 0 && ms.isFlushCacheRequired()) {
    
                this.clearLocalCache();
            }

            List list;
            try {
    
                ++this.queryStack;
                //根据获取到的CacheKey ,从一级缓存中获取,查看是否存在此数据
                list = resultHandler == null ? (List)this.localCache.getObject(key) : null;
                if (list != null) {
    
                	//说明一级缓存中存在CacheKey的缓存数据:
                	//       直接调用handleLocallyCachedOutputParameters方法,返回缓存中的数据
                    this.handleLocallyCachedOutputParameters(ms, key, parameter, boundSql);
                } else {
    
                    //说明一级缓存中不存在CacheKey的缓存数据:
                	//       调用queryFromDatabase方法,查询数据库中的数据
                    list = this.queryFromDatabase(ms, parameter, rowBounds, resultHandler, key, boundSql);
                }
            } finally {
    
                --this.queryStack;
            }

            if (this.queryStack == 0) {
    
                Iterator var8 = this.deferredLoads.iterator();

                while(var8.hasNext()) {
    
                    BaseExecutor.DeferredLoad deferredLoad = (BaseExecutor.DeferredLoad)var8.next();
                    deferredLoad.load();
                }

                this.deferredLoads.clear();
                if (this.configuration.getLocalCacheScope() == LocalCacheScope.STATEMENT) {
    
                    this.clearLocalCache();
                }
            }

            return list;
        }
    }
    

this.queryFromDatabase(ms, parameter, rowBounds, resultHandler, key, boundSql)方法的源码,如下:


    private <E> List<E> queryFromDatabase(MappedStatement ms, Object parameter, RowBounds rowBounds, ResultHandler resultHandler, CacheKey key, BoundSql boundSql) throws SQLException {
    
        this.localCache.putObject(key, ExecutionPlaceholder.EXECUTION_PLACEHOLDER);

        List list;
        try {
    
        	//执行doQuery方法,从数据库中查询出数据存储到list集合中
            list = this.doQuery(ms, parameter, rowBounds, resultHandler, boundSql);
        } finally {
    
            this.localCache.removeObject(key);
        }
		//key:CacheKey   list:数据库中查询到的数据
		//根据查询结果,将数据库中的数据保存到缓存中
        this.localCache.putObject(key, list);
        if (ms.getStatementType() == StatementType.CALLABLE) {
    
            this.localOutputParameterCache.putObject(key, parameter);
        }

        return list;
    }
    

三. 二级缓存

3.1 二级缓存演示

⼆级缓存的原理和⼀级缓存原理⼀样,第⼀次查询,会将数据放⼊缓存中,然后第⼆次查询则会直接去缓存中取。但是⼀级缓存是基于sqlSession的,⽽⼆级缓存是基于mapper⽂件的namespace的,也就是说多个sqlSession可以共享⼀个mapper中的⼆级缓存区域,并且如果两个mapper的namespace 相同,即使是两个mapper,那么这两个mapper中执⾏sql查询到的数据也将存在相同的⼆级缓存区域中
在这里插入图片描述
大家知道,Mybatis默认二级缓存是关闭的,如果我们想在SpringBoot中打开二级缓存,只需要2步:

1.在application.properties中加上以下配置

mybatis.configuration.cache-enabled=true

2.在mapper的xml文件中的namespace中加上

<cache></cache>

即可。直接这样执行sql会报错,因为开启了二级缓存后entity类必须要序列化,序列化后就可以正常使用了。

这里需要注意的是,同样的sql多次执行中控制台还是会显示sql语句,但是这并不是说发了2次sql,因为如果缓存生效了,控制台里会提示Cache Hit xxx,就表明是从缓存中取得的结果了

在这里插入图片描述
我们可以看到Cache Hit xxx:0.5,这代表在二级缓存中命中了数据,即mapper2查询出来的数据是来自缓存中,
但是此时有一个问题:为什么user1 != user2 呢?结果为 false
这是因为二级缓存与一级缓存不同:一级缓存缓存的是对象,二级缓存缓存的是数据;在命中二级缓存时,会重新封装成一个对象返回,因此二者是false。


3.2 标签 < cache/> 的解析

⼆级缓存构建在⼀级缓存之上,在收到查询请求时,MyBatis ⾸先会查询⼆级缓存,若⼆级缓存未命中,再去查询⼀级缓存,⼀级缓存没有,再查询数据库。

数据查询顺序:⼆级缓存 -----> ⼀级缓存 -----> 数据库

与⼀级缓存不同,⼆级缓存和具体的命名空间绑定,⼀个Mapper中有⼀个Cache相同Mapper中的MappedStatement共⽤⼀个Cache,⼀级缓存则是和 SqlSession 绑定。

我们都知道 < cache/>标签存在于mapper.xml文件中,SqlMapConfig.xml文件中引入了相应的mapper.xml文件,所以我们需要从SqlMapConfig.xml的文件解析开始探究标签< cache/> 的解析:
在这里插入图片描述


由上图可知,SqlSessionFactoryBuilder().build(resourceAsStream)方法会对配置文件的进行解析,因此我们点进去查看相应的源码:
在这里插入图片描述


继续点击 this.build()方法:
在这里插入图片描述


XML的解析⼯作主要交给XMLConfigBuilder.parse()⽅法来实现,XMLConfigBuilder这个类是Mybatis专门用来解析SqlMapConfig.xml配置文件的类,我们点击parser.parse()方法查看具体的解析细节:
在这里插入图片描述


点击XMLConfigBuilder.parse()⽅法后,我们发现是调用了XMLConfigBuilder类中的parse()方法;并且再次调用了本类中的parseConfiguration()方法传递了根标签this.parser.evalNode("/configuration")的内容(也就是下图配置文件里面的内容)
在这里插入图片描述


继续点击this.parseConfiguration(this.parser.evalNode("/configuration"))方法

    private void parseConfiguration(XNode root) {
    
        try {
    
        
			//部分源代码省略。。。主要用于解析其他标签
            
			//解析mapper标签
            this.mapperElement(root.evalNode("mappers"));
            
        } catch (Exception var3) {
    
            throw new BuilderException("Error parsing SQL Mapper Configuration. Cause: " + var3, var3);
        }
    }

我们主要关注解析mapper.xml文件的代码,点击this.mapperElement(root.evalNode("mappers"))方法

    private void mapperElement(XNode parent) throws Exception {
    
        if (parent != null) {
    
            Iterator var2 = parent.getChildren().iterator();

            while(true) {
    
                while(var2.hasNext()) {
    
                    XNode child = (XNode)var2.next();
                    String resource;
                    if ("package".equals(child.getName())) {
    
                        resource = child.getStringAttribute("name");
                        this.configuration.addMappers(resource);
                    } else {
    
                        resource = child.getStringAttribute("resource");
                        String url = child.getStringAttribute("url");
                        String mapperClass = child.getStringAttribute("class");
                        XMLMapperBuilder mapperParser;
                        InputStream inputStream;
                        if (resource != null && url == null && mapperClass == null) {
    
                            ErrorContext.instance().resource(resource);
                            inputStream = Resources.getResourceAsStream(resource);

                            //创建XMLMapperBuilder对象
                            mapperParser = new XMLMapperBuilder(inputStream, this.configuration, resource, this.configuration.getSqlFragments());
                            //⽣成XMLMapperBuilder,并执⾏其parse⽅法
                            mapperParser.parse();

                        } else if (resource == null && url != null && mapperClass == null) {
    
                            ErrorContext.instance().resource(url);
                            inputStream = Resources.getUrlAsStream(url);
                            mapperParser = new XMLMapperBuilder(inputStream, this.configuration, url, this.configuration.getSqlFragments());
                            mapperParser.parse();
                        } else {
    
                            if (resource != null || url != null || mapperClass == null) {
    
                                throw new BuilderException("A mapper element may only specify a url, resource or class, but not more than one.");
                            }

                            Class<?> mapperInterface = Resources.classForName(mapperClass);
                            this.configuration.addMapper(mapperInterface);
                        }
                    }
                }

                return;
            }
        }
    }

这里我们主要注意XMLMapperBuilder这个类,XMLMapperBuilder是Mybatis专门用来用来解析mapper.xml文件的


继续点击 mapperParser.parse()方法,查看mapper文件的解析流程
在这里插入图片描述


继续点击XMLMapperBuilder类中的this.configurationElement(this.parser.evalNode("/mapper"))方法,查看mapper文件的解析流程(这里和XMLConfigBuilder中的方法类似)

private void configurationElement(XNode context) {
    
        try {
    
            String namespace = context.getStringAttribute("namespace");
           if (namespace != null && !namespace.isEmpty()) {
    
                //部分源代码省略。。。
                
				// 最终在这⾥看到了关于cache属性的处理
                this.cacheElement(context.evalNode("cache"));
                
                //部分源代码省略。。。
                         
				//解析<select /> <insert /> <update /> <delete >节点们
				//这里会将生成的Cache包装到对应的MappedStatement
				this.buildStatementFromContext(context.evalNodes("select|insert|update|delete"));

            } else {
    
                throw new BuilderException("Mapper's namespace cannot be empty");
            }
        } catch (Exception var3) {
    
            throw new BuilderException("Error parsing Mapper XML. The XML location is '" + this.resource + "'. Cause: " + var3, var3);
        }
    }

在这里插入图片描述


我们进去 this.cacheElement(context.evalNode("cache"))方法查看

        if (context != null) {
    
        
            //解析<cache/>标签的type属性,这⾥我们可以⾃定义cache的实现类,⽐如redisCache,如果没有⾃定义,这⾥使⽤和⼀级缓存相同的PERPETUAL
            String type = context.getStringAttribute("type", "PERPETUAL");
            
            Class<? extends Cache> typeClass = this.typeAliasRegistry.resolveAlias(type);
            String eviction = context.getStringAttribute("eviction", "LRU");
            Class<? extends Cache> evictionClass = this.typeAliasRegistry.resolveAlias(eviction);
            Long flushInterval = context.getLongAttribute("flushInterval");
            Integer size = context.getIntAttribute("size");
            boolean readWrite = !context.getBooleanAttribute("readOnly", false);
            boolean blocking = context.getBooleanAttribute("blocking", false);
            Properties props = context.getChildrenAsProperties();
            
			// 构建Cache对象
            this.builderAssistant.useNewCache(typeClass, evictionClass, flushInterval, size, readWrite, blocking, props);

        }

    }

来看看是如何构建Cache对象的:MapperBuilderAssistant类中的useNewCache()方法

    public Cache useNewCache(Class<? extends Cache> typeClass, Class<? extends Cache> evictionClass, Long flushInterval, Integer size, boolean readWrite, boolean blocking, Properties props) {
    
    	// 1.⽣成Cache对象
        Cache cache = (new CacheBuilder(this.currentNamespace))
        //这⾥如果我们定义了<cache/>中的type,就使⽤⾃定义的Cache,否则使⽤和⼀级缓存相同的PerpetualCache
        .implementation((Class)this.valueOrDefault(typeClass, PerpetualCache.class))
        .addDecorator((Class)this.valueOrDefault(evictionClass, LruCache.class))
        .clearInterval(flushInterval)
        .size(size)
        .readWrite(readWrite)
        .blocking(blocking)
        .properties(props)
        .build();
        // 2.添加到Configuration中
        this.configuration.addCache(cache);
        // 3.并将cache赋值给MapperBuilderAssistant.currentCache
        this.currentCache = cache;
        return cache;
    }

我们看到⼀个Mapper.xml只会解析⼀次标签,也就是只创建⼀次Cache对象,放进configuration中, 并将cache赋值给MapperBuilderAssistant.currentCache

这就对应了我们前面所说的一个Mapper对应一个Cache对象
在这里插入图片描述

那么接下来,我们看看另外一句话是什么意思:相同的Mapper中的Mappedstatement公用一个Cache?
在这里插入图片描述
我们回到解析完< cache/> 标签的方法,XMLMapperBuilder类中的configurationElement方法

private void configurationElement(XNode context) {
    
        try {
    
            String namespace = context.getStringAttribute("namespace");
            if (namespace != null && !namespace.isEmpty()) {
    
                //部分源代码省略。。。
                
				// 最终在这⾥看到了关于cache属性的处理
                this.cacheElement(context.evalNode("cache"));
                
                //部分源代码省略。。。
                         
				//解析<select /> <insert /> <update /> <delete >节点们
				//这里会将生成的Cache包装到对应的MappedStatement
				this.buildStatementFromContext(context.evalNodes("select|insert|update|delete"));

            } else {
    
                throw new BuilderException("Mapper's namespace cannot be empty");
            }
        } catch (Exception var3) {
    
            throw new BuilderException("Error parsing Mapper XML. The XML location is '" + this.resource + "'. Cause: " + var3, var3);
        }
    }

重点关注buildStatementFromContext(context.evalNodes("select|insert|update|delete"));这个方法
这个方法主要是做两件事:
1,解析<select /> <insert /> <update /> <delete >节点们
2,生成的Cache包装到对应的MappedStatement


我们点击buildStatementFromContext(context.evalNodes("select|insert|update|delete"))这个方法看一下具体实现
buildStatementFromContext(context.evalNodes(“select|insert|update|delete”));将Cache包装到MappedStatement
在这里插入图片描述


继续点击this.buildStatementFromContext(list, this.configuration.getDatabaseId())方法
XMLMapperBuilder类中的buildStatementFromContext方法

    private void buildStatementFromContext(List<XNode> list, String requiredDatabaseId) {
    
        Iterator var3 = list.iterator();

        while(var3.hasNext()) {
    
            XNode context = (XNode)var3.next();
            //创建XMLStatementBuilder 对象,执行解析
            XMLStatementBuilder statementParser = new XMLStatementBuilder(this.configuration, this.builderAssistant, context, requiredDatabaseId);
            try {
    
                //每一条执行语句转换成一个MappedStatement
                statementParser.parseStatementNode();
            } catch (IncompleteElementException var7) {
    
            	//解析失败,添加到configuration 中
                this.configuration.addIncompleteStatement(statementParser);
            }
        }

    }

继续点击 statementParser.parseStatementNode()方法

    public void parseStatementNode() {
    
        String id = this.context.getStringAttribute("id");
        String databaseId = this.context.getStringAttribute("databaseId");
        if (this.databaseIdMatchesCurrent(id, databaseId, this.requiredDatabaseId)) {
    
            
            //部分源代码省略。。。主要是解析标签内相应的属性
			
			//创建MappedStatement对象
            this.builderAssistant.addMappedStatement(id, sqlSource, statementType, sqlCommandType, fetchSize, timeout, parameterMap, parameterTypeClass, resultMap, resultTypeClass, resultSetTypeEnum, flushCache, useCache, resultOrdered, (KeyGenerator)keyGenerator, keyProperty, keyColumn, databaseId, langDriver, resultSets);
        }
    }

继续点击this.builderAssistant.addMappedStatement(0方法

    public MappedStatement addMappedStatement(String id, SqlSource sqlSource, StatementType statementType, SqlCommandType sqlCommandType, Integer fetchSize, Integer timeout, String parameterMap, Class<?> parameterType, String resultMap, Class<?> resultType, ResultSetType resultSetType, boolean flushCache, boolean useCache, boolean resultOrdered, KeyGenerator keyGenerator, String keyProperty, String keyColumn, String databaseId, LanguageDriver lang, String resultSets) {
    
        if (this.unresolvedCacheRef) {
    
            throw new IncompleteElementException("Cache-ref not yet resolved");
        } else {
    
            id = this.applyCurrentNamespace(id, false);
            boolean isSelect = sqlCommandType == SqlCommandType.SELECT;


            MappedStatement.Builder statementBuilder = (new MappedStatement
            				.Builder(this.configuration, id, sqlSource, sqlCommandType))
            				.resource(this.resource)
            				.fetchSize(fetchSize)
            				.timeout(timeout)
            				.statementType(statementType)
            				.keyGenerator(keyGenerator)
            				.keyProperty(keyProperty)
            				.keyColumn(keyColumn)
            				.databaseId(databaseId)
            				.lang(lang)
            				.resultOrdered(resultOrdered)
            				.resultSets(resultSets)
            				.resultMaps(this.getStatementResultMaps(resultMap, resultType, id))
            				.resultSetType(resultSetType)
            				.flushCacheRequired((Boolean)this.valueOrDefault(flushCache, !isSelect))
            				.useCache((Boolean)this.valueOrDefault(useCache, isSelect))
            				// 在这⾥将之前⽣成的Cache封装到MappedStatement
            				.cache(this.currentCache);

            ParameterMap statementParameterMap = this.getStatementParameterMap(parameterMap, parameterType, id);
            if (statementParameterMap != null) {
    
                statementBuilder.parameterMap(statementParameterMap);
            }

            MappedStatement statement = statementBuilder.build();
            this.configuration.addMappedStatement(statement);
            return statement;
        }
    }

我们知道每一个mapper.xml文件中的<select /> <insert /> <update /> <delete >标签都是一个MappedStatement对象,并且我们在刚刚的源码中发现,在执行 MappedStatement.Builder创建MappedStatement对象时候,会执行.cache(this.currentCache)方法,将之前解析的Mapper.xml时创建的cache封装到对应的MappedStatement对象中
在这里插入图片描述

我们看到将Mapper中创建的Cache对象,加⼊到了每个MappedStatement对象中,也就是同⼀个
Mapper中所有的MappedStatement中共用的是一个Cache对象

这就对应了另外一句话:相同的Mapper中的Mappedstatement公用一个Cache
在这里插入图片描述

到此为止,< cache/> 的解析流程就全部结束了。


3.3 查询源码分析

具体流程与一级缓存相似,以流程图的形式展示:
在这里插入图片描述

主要的区别在于:在我们开启二级缓存以后,Executor接口的抽象方法query()走的实现类为CachingExecutor,而不是一级缓存时候BaseExecutor。
在这里插入图片描述
通过查阅资料我们知道:CachingExecutor是一个缓存装饰类,CachingExecutor中包含了对BaseExecutor的引用


我们点击CachingExecutor类中的query()方法查看具体实现

    public <E> List<E> query(MappedStatement ms, Object parameterObject, RowBounds rowBounds, ResultHandler resultHandler) throws SQLException {
    
    	//获得BoundSql对象
        BoundSql boundSql = ms.getBoundSql(parameterObject);
        //创建 CacheKey  对象
        CacheKey key = this.createCacheKey(ms, parameterObject, rowBounds, boundSql);
        //查询
        return this.query(ms, parameterObject, rowBounds, resultHandler, key, boundSql);
    }

点击this.query(ms, parameterObject, rowBounds, resultHandler, key, boundSql)方法查看具体实现

    public <E> List<E> query(MappedStatement ms, Object parameterObject, RowBounds rowBounds, ResultHandler resultHandler, CacheKey key, BoundSql boundSql) throws SQLException {
    
        //从 MappedStatement 中获取Cache,注意这里的Cache是从MappedStatement中获取的
        //也就是我们上面解析Mapper中<cache/>标签中创建的,它保存在Configration中
        //我们在初始化解析xml时分析过每一个MappedStatement都有一个Cache对象,就是这里
        Cache cache = ms.getCache();
        if (cache != null) {
    
        	 //如果需要刷新缓存的话就刷新:flushCache="true"
        	 //判断标签中是否配置了刷新缓存的属性
            this.flushCacheIfRequired(ms);
            if (ms.isUseCache() && resultHandler == null) {
    
                this.ensureNoOutParams(ms, boundSql);
                //从二级缓存中,获取结果
                List<E> list = (List)this.tcm.getObject(cache, key);
                if (list == null) {
    
                	//如果没有值,则执行查询,这个查询也是先走一级缓存,以及缓存也没有的话,则进行查询数据库
                    list = this.delegate.query(ms, parameterObject, rowBounds, resultHandler, key, boundSql);
                    this.tcm.putObject(cache, key, list);
                }

                return list;
            }
        }

如上,注意⼆级缓存是从 MappedStatement 中获取的。由于 MappedStatement 存在于全局配置
中,可以多个 CachingExecutor 获取到,这样就会出现线程安全问题。除此之外,若不加以控制,多个
事务共⽤⼀个缓存实例,会导致脏读问题。⾄于脏读问题,需要借助其他类来处理,也就是上⾯代码中
tcm 变量对应的类型。下⾯分析⼀下。
TransactionalCache

public class TransactionalCache implements Cache {
    
	 //真正的缓存对象,和上⾯的Map<Cache, TransactionalCache>中的Cache是同⼀个
	 private final Cache delegate;
	 private boolean clearOnCommit;
	 // 在事务被提交前,所有从数据库中查询的结果将缓存在此集合中
	 private final Map<Object, Object> entriesToAddOnCommit;
	 // 在事务被提交前,当缓存未命中时,CacheKey 将会被存储在此集合中
	 private final Set<Object> entriesMissedInCache;
	 @Override
	 public Object getObject(Object key) {
    
	 // 查询的时候是直接从delegate中去查询的,也就是从真正的缓存对象中查询
	 Object object = delegate.getObject(key);
	 
	if (object == null) {
    
		 // 缓存未命中,则将 key 存⼊到 entriesMissedInCache 中
		 entriesMissedInCache.add(key);
	 }
	 if (clearOnCommit) {
    
		 return null;
	 } else {
    
	 	return object;
	 }

	 @Override
	 public void putObject(Object key, Object object) {
    
	 // 将键值对存⼊到 entriesToAddOnCommit 这个Map中中,⽽⾮真实的缓存对象delegate 中
	 entriesToAddOnCommit.put(key, object);
	 }
	 
	 @Override
	 public Object removeObject(Object key) {
    
	 	return null;
	 }
	 
	 @Override
	 public void clear() {
    
		 clearOnCommit = true;
		 // 清空 entriesToAddOnCommit,但不清空 delegate 缓存
		 entriesToAddOnCommit.clear();
	 }
	 
	 public void commit() {
    
	 	// 根据 clearOnCommit 的值决定是否清空 delegate
	 	if (clearOnCommit) {
    
		 delegate.clear();
		 }
		 // 刷新未缓存的结果到 delegate 缓存中
		 flushPendingEntries();
		 // 重置 entriesToAddOnCommit 和 entriesMissedInCache
		 reset();
	 }
	 public void rollback() {
    
		 unlockMissedEntries();
		 reset();
	 }
	 private void reset() {
    
		 clearOnCommit = false;
		 // 清空集合
		 entriesToAddOnCommit.clear();
		 entriesMissedInCache.clear();
	 }
	 private void flushPendingEntries() {
    
		 for (Map.Entry<Object, Object> entry :
			entriesToAddOnCommit.entrySet()) {
    
				 // 将 entriesToAddOnCommit 中的内容转存到 delegate 中
				 delegate.putObject(entry.getKey(), entry.getValue());
			 }
			 for (Object entry : entriesMissedInCache) {
    
				 if (!entriesToAddOnCommit.containsKey(entry)) {
    
				 // 存⼊空值
				 delegate.putObject(entry, null);
			 }
		 }
	 }
	 private void unlockMissedEntries() {
    
		 for (Object entry : entriesMissedInCache) {
    
			 try {
    
				 // 调⽤ removeObject 进⾏解锁
				 delegate.removeObject(entry);
			 } catch (Exception e) {
    
			 	log.warn("...");
			 }
		 }
	 }
}

存储⼆级缓存对象的时候是放到了TransactionalCache.entriesToAddOnCommit这个map中,但是每
次查询的时候是直接从TransactionalCache.delegate中去查询的,所以这个⼆级缓存查询数据库后,设
置缓存值是没有⽴刻⽣效的,主要是因为直接存到 delegate 会导致脏数据问题

注意:我们在这里还发现有一个delegate类执行了query方法,这个delegate类很陌生
在这里插入图片描述

我们将鼠标放置在delegate类上后发现,delegate类是SimpleExecutor类,而SimpleExecutor是BaseExecutor的具体实现类,所以delegate类执行了query方法,就是BaseExecutor执行query方法;也就是说二级缓存没有命中的情况下,会接着一级缓存中找,最后才会查询数据库。
在这里插入图片描述

最后,我们分析一下`tcm.putObject(cache, key, list)这个方法
先说结论:tcm.putObject()方法是先将数据库中查询出来的数据放在一个entriesToAddOnCommit的map集合中,只有在sqlsession进行commit操作的时候,才会将entriesToAddOnCommit的数据刷入二级缓存中

在这里插入图片描述

tcm.putObject()最终调用的是TransactionalCache类中的putObject()方法,如下图
在这里插入图片描述
在这里插入图片描述
在这里插入图片描述
在这里插入图片描述

这个时候,我们发现,我们存的时候是存在entriesToAddOnCommit中,但是我们取是直接从delegate这个缓存对象中取的,这就引出了为啥执行一个commit()操作,具体执行的源代码如下图:
在这里插入图片描述
flushPendingEntries()方法的具体实现
在这里插入图片描述

3.4 总结

在⼆级缓存的设计上,MyBatis⼤量地运⽤了装饰者模式,如CachingExecutor, 以及各种Cache接⼝的装饰器。

  • ⼆级缓存实现了Sqlsession之间的缓存数据共享,属于namespace级别
  • ⼆级缓存具有丰富的缓存策略。
  • ⼆级缓存可由多个装饰器,与基础缓存组合⽽成
  • ⼆级缓存⼯作由 ⼀个缓存装饰执⾏器CachingExecutor和 ⼀个事务型预缓存TransactionalCache完成。
版权声明:本文为博主原创文章,遵循 CC 4.0 BY-SA 版权协议,转载请附上原文出处链接和本声明。
本文链接:https://blog.csdn.net/weixin_44304847/article/details/124835665

智能推荐

FTP命令字和返回码_ftp 登录返回230-程序员宅基地

文章浏览阅读3.5k次,点赞2次,收藏13次。为了从FTP服务器下载文件,需要要实现一个简单的FTP客户端。FTP(文件传输协议) 是 TCP/IP 协议组中的应用层协议。FTP协议使用字符串格式命令字,每条命令都是一行字符串,以“\r\n”结尾。客户端发送格式是:命令+空格+参数+"\r\n"的格式服务器返回格式是以:状态码+空格+提示字符串+"\r\n"的格式,代码只要解析状态码就可以了。读写文件需要登陆服务器,特殊用..._ftp 登录返回230

centos7安装rabbitmq3.6.5_centos7 安装rabbitmq3.6.5-程序员宅基地

文章浏览阅读648次。前提:systemctl stop firewalld 关闭防火墙关闭selinux查看getenforce临时关闭setenforce 0永久关闭sed-i'/SELINUX/s/enforcing/disabled/'/etc/selinux/configselinux的三种模式enforcing:强制模式,SELinux 运作中,且已经正确的开始限制..._centos7 安装rabbitmq3.6.5

idea导入android工程,idea怎样导入Android studio 项目?-程序员宅基地

文章浏览阅读5.8k次。满意答案s55f2avsx2017.09.05采纳率:46%等级:12已帮助:5646人新版Android Studio/IntelliJ IDEA可以直接导入eclipse项目,不再推荐使用eclipse导出gradle的方式2启动Android Studio/IntelliJ IDEA,选择 import project3选择eclipse 项目4选择 create project f..._android studio 项目导入idea 看不懂安卓项目

浅谈AI大模型技术:概念、发展和应用_ai大模型应用开发-程序员宅基地

文章浏览阅读860次,点赞2次,收藏6次。AI大模型技术已经在自然语言处理、计算机视觉、多模态交互等领域取得了显著的进展和成果,同时也引发了一系列新的挑战和问题,如数据质量、计算效率、知识可解释性、安全可靠性等。城市运维涉及到多个方面,如交通管理、环境监测、公共安全、社会治理等,它们需要处理和分析大量的多模态数据,如图像、视频、语音、文本等,并根据不同的场景和需求,提供合适的决策和响应。知识搜索有多种形式,如语义搜索、对话搜索、图像搜索、视频搜索等,它们可以根据用户的输入和意图,从海量的数据源中检索出最相关的信息,并以友好的方式呈现给用户。_ai大模型应用开发

非常详细的阻抗测试基础知识_阻抗实部和虚部-程序员宅基地

文章浏览阅读8.2k次,点赞12次,收藏121次。为什么要测量阻抗呢?阻抗能代表什么?阻抗测量的注意事项... ...很多人可能会带着一系列的问题来阅读本文。不管是数字电路工程师还是射频工程师,都在关注各类器件的阻抗,本文非常值得一读。全文13000多字,认真读完大概需要2小时。一、阻抗测试基本概念阻抗定义:阻抗是元器件或电路对周期的交流信号的总的反作用。AC 交流测试信号 (幅度和频率)。包括实部和虚部。​图1 阻抗的定义阻抗是评测电路、元件以及制作元件材料的重要参数。那么什么是阻抗呢?让我们先来看一下阻抗的定义。首先阻抗是一个矢量。通常,阻抗是_阻抗实部和虚部

小学生python游戏编程arcade----基本知识1_arcade语言 like-程序员宅基地

文章浏览阅读955次。前面章节分享试用了pyzero,pygame但随着想增加更丰富的游戏内容,好多还要进行自己编写类,从今天开始解绍一个新的python游戏库arcade模块。通过此次的《连连看》游戏实现,让我对swing的相关知识有了进一步的了解,对java这门语言也有了比以前更深刻的认识。java的一些基本语法,比如数据类型、运算符、程序流程控制和数组等,理解更加透彻。java最核心的核心就是面向对象思想,对于这一个概念,终于悟到了一些。_arcade语言 like

随便推点

【增强版短视频去水印源码】去水印微信小程序+去水印软件源码_去水印机要增强版-程序员宅基地

文章浏览阅读1.1k次。源码简介与安装说明:2021增强版短视频去水印源码 去水印微信小程序源码网站 去水印软件源码安装环境(需要材料):备案域名–服务器安装宝塔-安装 Nginx 或者 Apachephp5.6 以上-安装 sg11 插件小程序已自带解析接口,支持全网主流短视频平台,搭建好了就能用注:接口是公益的,那么多人用解析慢是肯定的,前段和后端源码已经打包,上传服务器之后在配置文件修改数据库密码。然后输入自己的域名,进入后台,创建小程序,输入自己的小程序配置即可安装说明:上传源码,修改data/_去水印机要增强版

verilog进阶语法-触发器原语_fdre #(.init(1'b0) // initial value of register (1-程序员宅基地

文章浏览阅读557次。1. 触发器是FPGA存储数据的基本单元2. 触发器作为时序逻辑的基本元件,官方提供了丰富的配置方式,以适应各种可能的应用场景。_fdre #(.init(1'b0) // initial value of register (1'b0 or 1'b1) ) fdce_osc (

嵌入式面试/笔试C相关总结_嵌入式面试笔试c语言知识点-程序员宅基地

文章浏览阅读560次。本该是不同编译器结果不同,但是尝试了g++ msvc都是先计算c,再计算b,最后得到a+b+c是经过赋值以后的b和c参与计算而不是6。由上表可知,将q复制到p数组可以表示为:*p++=*q++,*优先级高,先取到对应q数组的值,然后两个++都是在后面,该行运算完后执行++。在电脑端编译完后会分为text data bss三种,其中text为可执行程序,data为初始化过的ro+rw变量,bss为未初始化或初始化为0变量。_嵌入式面试笔试c语言知识点

57 Things I've Learned Founding 3 Tech Companies_mature-程序员宅基地

文章浏览阅读2.3k次。57 Things I've Learned Founding 3 Tech CompaniesJason Goldberg, Betashop | Oct. 29, 2010, 1:29 PMI’ve been founding andhelping run techn_mature

一个脚本搞定文件合并去重,大数据处理,可以合并几个G以上的文件_python 超大文本合并-程序员宅基地

文章浏览阅读1.9k次。问题:先讲下需求,有若干个文本文件(txt或者csv文件等),每行代表一条数据,现在希望能合并成 1 个文本文件,且需要去除重复行。分析:一向奉行简单原则,如无必要,绝不复杂。如果数据量不大,那么如下两条命令就可以搞定合并:cat a.txt >> new.txtcat b.txt >> new.txt……去重:cat new...._python 超大文本合并

支付宝小程序iOS端过渡页DFLoadingPageRootController分析_类似支付宝页面过度加载页-程序员宅基地

文章浏览阅读489次。这个过渡页是第一次打开小程序展示的,点击某个小程序前把手机的开发者->network link conditioner->enable & very bad network 就会在停在此页。比如《支付宝运动》这个小程序先看这个类的.h可以看到它继承于DTViewController点击左上角返回的方法- (void)back;#import "DTViewController.h"#import "APBaseLoadingV..._类似支付宝页面过度加载页

推荐文章

热门文章

相关标签