成人国产在线小视频_日韩寡妇人妻调教在线播放_色成人www永久在线观看_2018国产精品久久_亚洲欧美高清在线30p_亚洲少妇综合一区_黄色在线播放国产_亚洲另类技巧小说校园_国产主播xx日韩_a级毛片在线免费

資訊專欄INFORMATION COLUMN

Spring Boot緩存源碼分析

wangbjun / 2260人閱讀

摘要:依賴配置文件設(shè)定緩存的默認數(shù)據(jù)過期策略應(yīng)用上加上注解然后就可以在代碼里面使用注解了,像這樣。使用該集合過濾執(zhí)行緩存處理。從中獲取實例,然后執(zhí)行放入緩存的操作是一個標準接口,其中就是的實現(xiàn)類。

前言

項目里面要增加一個應(yīng)用緩存,原本想著要怎么怎么來整合ehcache和springboot,做好準備配置這個配置那個,結(jié)果只需要做三件事:

pom依賴

寫好一個ehcache的配置文件

在boot的application上加上注解@EnableCaching.
這就完事了,是不是很魔幻。

pom依賴


            net.sf.ehcache
            ehcache
            2.10.5

配置文件



    
    

應(yīng)用上加上EnableCaching注解

@SpringBootApplication
@EnableCaching
public class EhCacheApplication {

    public static void main(String[] args) {
        SpringApplication.run(EhCacheApplication.class, args);
    }
}

然后就可以在代碼里面使用cache注解了,像這樣。

@CachePut(value = "fish-ehcache", key = "#person.id")
    public Person save(Person person) {
        System.out.println("為id、key為:" + person.getId() + "數(shù)據(jù)做了緩存");
        return person;
    }

    @CacheEvict(value = "fish-ehcache")
    public void remove(Long id) {
        System.out.println("刪除了id、key為" + id + "的數(shù)據(jù)緩存");
    }


    @Cacheable(value = "fish-ehcache", key = "#person.id")
    public Person findOne(Person person) {
        findCount.incrementAndGet();
        System.out.println("為id、key為:" + person.getId() + "數(shù)據(jù)做了緩存");
        return person;
    }

很方便對不對。下面,我們就來挖一挖,看看spring是怎么來做到的。主要分成兩部分,一是啟動的時候做了什么,二是運行的時候做了什么,三是和第三方緩存組件的適配

啟動的時候做了什么、

這個得從@EnableCaching標簽開始,在使用緩存功能時,在springboot的Application啟動類上需要添加注解@EnableCaching,這個標簽引入了

@Target({ElementType.TYPE})
@Retention(RetentionPolicy.RUNTIME)
@Documented
@Import({CachingConfigurationSelector.class})
public @interface EnableCaching {
    boolean proxyTargetClass() default false;
    AdviceMode mode() default AdviceMode.PROXY;
    int order() default 2147483647;
}

引入了CachingConfigurationSelector類,這個類便開啟了緩存功能的配置。這個類添加了AutoProxyRegistrar.java,ProxyCachingConfiguration.java兩個類。

AutoProxyRegistrar : 實現(xiàn)了ImportBeanDefinitionRegistrar接口。這里看不懂,還需要繼續(xù)學習。

ProxyCachingConfiguration : 是一個配置類,生成了BeanFactoryCacheOperationSourceAdvisor,CacheOperationSource,和CacheInterceptor這三個bean。

CacheOperationSource封裝了cache方法簽名注解的解析工作,形成CacheOperation的集合。CacheInterceptor使用該集合過濾執(zhí)行緩存處理。解析緩存注解的類是SpringCacheAnnotationParser,其主要方法如下

/**
由CacheOperationSourcePointcut作為注解切面,會解析
SpringCacheAnnotationParser.java
掃描方法簽名,解析被緩存注解修飾的方法,將生成一個CacheOperation的子類并將其保存到一個數(shù)組中去
**/
protected Collection parseCacheAnnotations(SpringCacheAnnotationParser.DefaultCacheConfig cachingConfig, AnnotatedElement ae) {
        Collection ops = null;
        //找@cacheable注解方法
        Collection cacheables = AnnotatedElementUtils.getAllMergedAnnotations(ae, Cacheable.class);
        if (!cacheables.isEmpty()) {
            ops = this.lazyInit(ops);
            Iterator var5 = cacheables.iterator();

            while(var5.hasNext()) {
                Cacheable cacheable = (Cacheable)var5.next();
                ops.add(this.parseCacheableAnnotation(ae, cachingConfig, cacheable));
            }
        }
        //找@cacheEvict注解的方法
        Collection evicts = AnnotatedElementUtils.getAllMergedAnnotations(ae, CacheEvict.class);
        if (!evicts.isEmpty()) {
            ops = this.lazyInit(ops);
            Iterator var12 = evicts.iterator();

            while(var12.hasNext()) {
                CacheEvict evict = (CacheEvict)var12.next();
                ops.add(this.parseEvictAnnotation(ae, cachingConfig, evict));
            }
        }
        //找@cachePut注解的方法
        Collection puts = AnnotatedElementUtils.getAllMergedAnnotations(ae, CachePut.class);
        if (!puts.isEmpty()) {
            ops = this.lazyInit(ops);
            Iterator var14 = puts.iterator();

            while(var14.hasNext()) {
                CachePut put = (CachePut)var14.next();
                ops.add(this.parsePutAnnotation(ae, cachingConfig, put));
            }
        }
        Collection cachings = AnnotatedElementUtils.getAllMergedAnnotations(ae, Caching.class);
        if (!cachings.isEmpty()) {
            ops = this.lazyInit(ops);
            Iterator var16 = cachings.iterator();

            while(var16.hasNext()) {
                Caching caching = (Caching)var16.next();
                Collection cachingOps = this.parseCachingAnnotation(ae, cachingConfig, caching);
                if (cachingOps != null) {
                    ops.addAll(cachingOps);
                }
            }
        }
        return ops;
}

解析Cachable,Caching,CachePut,CachEevict 這四個注解對應(yīng)的方法都保存到了Collection 集合中。

執(zhí)行方法時做了什么

執(zhí)行的時候,主要使用了CacheInterceptor類。

public class CacheInterceptor extends CacheAspectSupport implements MethodInterceptor, Serializable {
    public CacheInterceptor() {
    }

    public Object invoke(final MethodInvocation invocation) throws Throwable {
        Method method = invocation.getMethod();
        CacheOperationInvoker aopAllianceInvoker = new CacheOperationInvoker() {
            public Object invoke() {
                try {
                    return invocation.proceed();
                } catch (Throwable var2) {
                    throw new ThrowableWrapper(var2);
                }
            }
        };

        try {
            return this.execute(aopAllianceInvoker, invocation.getThis(), method, invocation.getArguments());
        } catch (ThrowableWrapper var5) {
            throw var5.getOriginal();
        }
    }
}

這個攔截器繼承了CacheAspectSupport類和MethodInterceptor接口。其中CacheAspectSupport封裝了主要的邏輯。比如下面這段。

/**
CacheAspectSupport.java
執(zhí)行@CachaEvict @CachePut @Cacheable的主要邏輯代碼
**/

private Object execute(final CacheOperationInvoker invoker, Method method, CacheAspectSupport.CacheOperationContexts contexts) {
        if (contexts.isSynchronized()) {
            CacheAspectSupport.CacheOperationContext context = (CacheAspectSupport.CacheOperationContext)contexts.get(CacheableOperation.class).iterator().next();
            if (this.isConditionPassing(context, CacheOperationExpressionEvaluator.NO_RESULT)) {
                Object key = this.generateKey(context, CacheOperationExpressionEvaluator.NO_RESULT);
                Cache cache = (Cache)context.getCaches().iterator().next();

                try {
                    return this.wrapCacheValue(method, cache.get(key, new Callable() {
                        public Object call() throws Exception {
                            return CacheAspectSupport.this.unwrapReturnValue(CacheAspectSupport.this.invokeOperation(invoker));
                        }
                    }));
                } catch (ValueRetrievalException var10) {
                    throw (ThrowableWrapper)var10.getCause();
                }
            } else {
                return this.invokeOperation(invoker);
            }
        } else {
            /**
            執(zhí)行@CacheEvict的邏輯,這里是當beforeInvocation為true時清緩存
            **/
            this.processCacheEvicts(contexts.get(CacheEvictOperation.class), true, CacheOperationExpressionEvaluator.NO_RESULT);
            //獲取命中的緩存對象
            ValueWrapper cacheHit = this.findCachedItem(contexts.get(CacheableOperation.class));
            List cachePutRequests = new LinkedList();
            if (cacheHit == null) {
                //如果沒有命中,則生成一個put的請求
                this.collectPutRequests(contexts.get(CacheableOperation.class), CacheOperationExpressionEvaluator.NO_RESULT, cachePutRequests);
            }


            Object cacheValue;
            Object returnValue;
            /**
                如果沒有獲得緩存對象,則調(diào)用業(yè)務(wù)方法獲得返回對象,hasCachePut會檢查exclude的情況
            **/
            if (cacheHit != null && cachePutRequests.isEmpty() && !this.hasCachePut(contexts)) {
                cacheValue = cacheHit.get();
                returnValue = this.wrapCacheValue(method, cacheValue);
            } else {
                
                returnValue = this.invokeOperation(invoker);
                cacheValue = this.unwrapReturnValue(returnValue);
            }

            this.collectPutRequests(contexts.get(CachePutOperation.class), cacheValue, cachePutRequests);
            Iterator var8 = cachePutRequests.iterator();

            while(var8.hasNext()) {
                CacheAspectSupport.CachePutRequest cachePutRequest = (CacheAspectSupport.CachePutRequest)var8.next();
                /**
                執(zhí)行cachePut請求,將返回對象放到緩存中
                **/
                cachePutRequest.apply(cacheValue);
            }
            /**
            執(zhí)行@CacheEvict的邏輯,這里是當beforeInvocation為false時清緩存
            **/
            this.processCacheEvicts(contexts.get(CacheEvictOperation.class), false, cacheValue);
            return returnValue;
        }
    }

上面的代碼片段比較核心,均是cache的內(nèi)容,對于aop的源碼,這里不詳細展開,應(yīng)該單起一篇文章進行研究。主要的類和接口都在spring的context中,org.springframework.cache包中。

和第三方緩存組件的適配

通過以上的分析,知道了spring cache功能的來龍去脈,下面需要分析的是,為什么只需要maven聲明一下依賴,spring boot 就可以自動就適配了.

在上面的執(zhí)行方法中,我們看到了cachePutRequest.apply(cacheValue) ,這里會操作緩存,CachePutRequest是CacheAspectSupport的內(nèi)部類。

private class CachePutRequest {
        private final CacheAspectSupport.CacheOperationContext context;
        private final Object key;
        public CachePutRequest(CacheAspectSupport.CacheOperationContext context, Object key) {
            this.context = context;
            this.key = key;
        }
        public void apply(Object result) {
            if (this.context.canPutToCache(result)) {
                //從context中獲取cache實例,然后執(zhí)行放入緩存的操作
                Iterator var2 = this.context.getCaches().iterator();
                while(var2.hasNext()) {
                    Cache cache = (Cache)var2.next();
                    CacheAspectSupport.this.doPut(cache, this.key, result);
                }
            }
        }
    }

Cache是一個標準接口,其中EhCacheCache就是EhCache的實現(xiàn)類。這里就是SpringBoot和Ehcache之間關(guān)聯(lián)的部分,那么context中的cache列表是什么時候生成的呢。答案是CacheAspectSupport的getCaches方法

protected Collection getCaches(CacheOperationInvocationContext context, CacheResolver cacheResolver) {
        Collection caches = cacheResolver.resolveCaches(context);
        if (caches.isEmpty()) {
            throw new IllegalStateException("No cache could be resolved for "" + context.getOperation() + "" using resolver "" + cacheResolver + "". At least one cache should be provided per cache operation.");
        } else {
            return caches;
        }
    }

而獲取cache是在每一次進行進行緩存操作的時候執(zhí)行??梢钥匆幌抡{(diào)用棧

貌似有點跑題,拉回來... 在spring-boot-autoconfigure包里,有所有自動裝配相關(guān)的類。這里有個EhcacheCacheConfiguration類 ,如下

@Configuration
@ConditionalOnClass({Cache.class, EhCacheCacheManager.class})
@ConditionalOnMissingBean({CacheManager.class})
@Conditional({CacheCondition.class, EhCacheCacheConfiguration.ConfigAvailableCondition.class})
class EhCacheCacheConfiguration {
 ......
 static class ConfigAvailableCondition extends ResourceCondition {
        ConfigAvailableCondition() {
            super("EhCache", "spring.cache.ehcache", "config", new String[]{"classpath:/ehcache.xml"});
        }
    }    
}

這里會直接判斷類路徑下是否有ehcache.xml文件

文章版權(quán)歸作者所有,未經(jīng)允許請勿轉(zhuǎn)載,若此文章存在違規(guī)行為,您可以聯(lián)系管理員刪除。

轉(zhuǎn)載請注明本文地址:http://systransis.cn/yun/72159.html

相關(guān)文章

  • spring boot - 收藏集 - 掘金

    摘要:引入了新的環(huán)境和概要信息,是一種更揭秘與實戰(zhàn)六消息隊列篇掘金本文,講解如何集成,實現(xiàn)消息隊列。博客地址揭秘與實戰(zhàn)二數(shù)據(jù)緩存篇掘金本文,講解如何集成,實現(xiàn)緩存。 Spring Boot 揭秘與實戰(zhàn)(九) 應(yīng)用監(jiān)控篇 - HTTP 健康監(jiān)控 - 掘金Health 信息是從 ApplicationContext 中所有的 HealthIndicator 的 Bean 中收集的, Spring...

    rollback 評論0 收藏0
  • Java學習路線總結(jié),搬磚工逆襲Java架構(gòu)師(全網(wǎng)最強)

    摘要:哪吒社區(qū)技能樹打卡打卡貼函數(shù)式接口簡介領(lǐng)域優(yōu)質(zhì)創(chuàng)作者哪吒公眾號作者架構(gòu)師奮斗者掃描主頁左側(cè)二維碼,加入群聊,一起學習一起進步歡迎點贊收藏留言前情提要無意間聽到領(lǐng)導們的談話,現(xiàn)在公司的現(xiàn)狀是碼農(nóng)太多,但能獨立帶隊的人太少,簡而言之,不缺干 ? 哪吒社區(qū)Java技能樹打卡?【打卡貼 day2...

    Scorpion 評論0 收藏0
  • 【推薦】最新200篇:技術(shù)文章整理

    摘要:作為面試官,我是如何甄別應(yīng)聘者的包裝程度語言和等其他語言的對比分析和主從復制的原理詳解和持久化的原理是什么面試中經(jīng)常被問到的持久化與恢復實現(xiàn)故障恢復自動化詳解哨兵技術(shù)查漏補缺最易錯過的技術(shù)要點大掃盲意外宕機不難解決,但你真的懂數(shù)據(jù)恢復嗎每秒 作為面試官,我是如何甄別應(yīng)聘者的包裝程度Go語言和Java、python等其他語言的對比分析 Redis和MySQL Redis:主從復制的原理詳...

    BicycleWarrior 評論0 收藏0

發(fā)表評論

0條評論

wangbjun

|高級講師

TA的文章

閱讀更多
最新活動
閱讀需要支付1元查看
<