Oscache的强行更新机制

时间:2022-01-23 22:20:15
背景 :
在产品中也许不需要强行更新,但是测试的时候往往需要。

part 1
当你强行更新缓存时会发生如下步骤:
step1)
GeneralCacheAdministrator.flushAll----->

step2)
Cache.flushAll(Date date, String origin)

flushAll的源代码如下:
public void flushAll(Date date, String origin) {
        //更新Cache的flushDateTime
        flushDateTime = date;
        //通知监听器更新事件,如果你没有注册监听器,这个方法没用
        dispatchCachewideEvent(CachewideEventType.CACHE_FLUSHED, date, origin);
    }

也就是说强行更新时仅仅只是修改Cache的flushDateTime 属性
并不是更改数据

part 2
接着你刷新页面看你的强行刷新时候有效果
step1)
Cache.getFromCache(String key, int refreshPeriod, String cronExpiry)---->
step2)
Cache.isStale(cacheEntry, refreshPeriod, cronExpiry)----->

看看isStale源代码:
protected boolean isStale(CacheEntry cacheEntry, int refreshPeriod, String cronExpiry) {
        boolean result = cacheEntry.needsRefresh(refreshPeriod) || isFlushed(cacheEntry);


       //cronExpiry 定义了绝对时间,通常你不会用,所以以下代码没用
        if ((cronExpiry != null) && (cronExpiry.length() > 0)) {
            try {
                FastCronParser parser = new FastCronParser(cronExpiry);
                result = result || parser.hasMoreRecentMatch(cacheEntry.getLastUpdate());
            } catch (ParseException e) {
                log.warn(e);
            }
        }

        return result;
    }

step3)
Cache.isFlushed(CacheEntry cacheEntry)

看看isFlushed源代码:
 public boolean isFlushed(CacheEntry cacheEntry) {
        if (flushDateTime != null) {
            long lastUpdate = cacheEntry.getLastUpdate();

            return ( flushDateTime.getTime() >= lastUpdate);
        } else {
            return false;
        }
    }

从上面源代码看出,因为Cache的 flushDateTime被更新,所以肯定会大于等于cacheEntry的最后更新时间,所以Cache会认为缓存的数据是stale的,于是从数据库重新取数据。


总而言之:强行更新只是做个标记,并不真正的获取新数据,下次从缓存读数据时,缓存会根据做的标记从数据更新数据,即使还没到更新的时候。