spring-aop

spring AOP 源碼淺析

概述

AbstractAutoProxyCreator通過postProcessAfterInitialization實現(xiàn)AOP功能藤乙。

源碼部分

  • AbstractAutoProxyCreator

    @Override
    public Object postProcessAfterInitialization(@Nullable Object bean, String beanName) throws BeansException {
        if (bean != null) {
            Object cacheKey = getCacheKey(bean.getClass(), beanName);
            if (!this.earlyProxyReferences.contains(cacheKey)) {
                return wrapIfNecessary(bean, beanName, cacheKey);
            }
        }
        return bean;
    }

    protected Object wrapIfNecessary(Object bean, String beanName, Object cacheKey) {
        if (StringUtils.hasLength(beanName) && this.targetSourcedBeans.contains(beanName)) {
            return bean;
        }
        if (Boolean.FALSE.equals(this.advisedBeans.get(cacheKey))) {
            return bean;
        }
        if (isInfrastructureClass(bean.getClass()) || shouldSkip(bean.getClass(), beanName)) {
            this.advisedBeans.put(cacheKey, Boolean.FALSE);
            return bean;
        }

        // getAdvicesAndAdvisorsForBean 由子類重寫
        Object[] specificInterceptors = getAdvicesAndAdvisorsForBean(bean.getClass(), beanName, null);
        if (specificInterceptors != DO_NOT_PROXY) {
            this.advisedBeans.put(cacheKey, Boolean.TRUE);
            Object proxy = createProxy(
                    bean.getClass(), beanName, specificInterceptors, new SingletonTargetSource(bean));
            this.proxyTypes.put(cacheKey, proxy.getClass());
            return proxy;
        }

        this.advisedBeans.put(cacheKey, Boolean.FALSE);
        return bean;
    }
  • 子類 AbstractAdvisorAutoProxyCreator 中的 getAdvicesAndAdvisorsForBean
    @Override
    @Nullable
    protected Object[] getAdvicesAndAdvisorsForBean(
            Class<?> beanClass, String beanName, @Nullable TargetSource targetSource) {

        List<Advisor> advisors = findEligibleAdvisors(beanClass, beanName);
        if (advisors.isEmpty()) {
            return DO_NOT_PROXY;
        }
        return advisors.toArray();
    }

    protected List<Advisor> findEligibleAdvisors(Class<?> beanClass, String beanName) {
        // BeanFacrory 中所有 Advisor 的實現(xiàn) 
        List<Advisor> candidateAdvisors = findCandidateAdvisors();
        // 有資格的 Advisor
        List<Advisor> eligibleAdvisors = findAdvisorsThatCanApply(candidateAdvisors, beanClass, beanName);
        extendAdvisors(eligibleAdvisors);
        if (!eligibleAdvisors.isEmpty()) {
            eligibleAdvisors = sortAdvisors(eligibleAdvisors);
        }
        return eligibleAdvisors;
    }

    protected List<Advisor> findCandidateAdvisors() {
        Assert.state(this.advisorRetrievalHelper != null, "No BeanFactoryAdvisorRetrievalHelper available");
        return this.advisorRetrievalHelper.findAdvisorBeans();
    }

    protected List<Advisor> findAdvisorsThatCanApply(
            List<Advisor> candidateAdvisors, Class<?> beanClass, String beanName) {

        ProxyCreationContext.setCurrentProxiedBeanName(beanName);
        try {
            return AopUtils.findAdvisorsThatCanApply(candidateAdvisors, beanClass);
        }
        finally {
            ProxyCreationContext.setCurrentProxiedBeanName(null);
        }
    }
  • 關于createProxy

    • ProxyFactory 對象中有要代理的bean和這個Bean上的advisor

    • Bean使用哪種代理

      1. 當Bean實現(xiàn)接口時别垮,Spring就會用JDK的動態(tài)代理蜡饵。
      2. 當Bean沒有實現(xiàn)接口時囤捻,Spring會自動使用CGlib實現(xiàn)湃交,但是前提是項目中導入了CGlib的相關依賴,否則Spring只能使用JDK來代理那些沒有實現(xiàn)接口的類坟漱,這樣生成的代理類會報錯。
    • AopProxy有兩個實現(xiàn)類JdkDynamicAopProxyCglibAopProxy更哄。都是構造 ReflectiveMethodInvocation.proceed()芋齿。

      • JdkDynamicAopProxy

            invocation = new ReflectiveMethodInvocation(proxy, target, method, args, targetClass, chain);
            retVal = invocation.proceed();
        
      • CglibAopProxy

            // CglibMethodInvocation 繼承于 ReflectiveMethodInvocation
            retVal = new CglibMethodInvocation(proxy, target, method, args, targetClass, chain, methodProxy).proceed();
        
  • ReflectiveMethodInvocation.proceed()

    public Object proceed() throws Throwable {
        // 當所有攔截器都執(zhí)行后,調(diào)用目標類的目標方法
        if (this.currentInterceptorIndex == this.interceptorsAndDynamicMethodMatchers.size() - 1) {
            return invokeJoinpoint();
        }

        Object interceptorOrInterceptionAdvice =
                this.interceptorsAndDynamicMethodMatchers.get(++this.currentInterceptorIndex);
        if (interceptorOrInterceptionAdvice instanceof InterceptorAndDynamicMethodMatcher) {
            // 動態(tài)攔截器
            InterceptorAndDynamicMethodMatcher dm =
                    (InterceptorAndDynamicMethodMatcher) interceptorOrInterceptionAdvice;
            if (dm.methodMatcher.matches(this.method, this.targetClass, this.arguments)) {
                return dm.interceptor.invoke(this);
            }
            else {
                // Dynamic matching failed.
                // Skip this interceptor and invoke the next in the chain.
                return proceed();
            }
        }
        else {
            // MethodInterceptor的實現(xiàn)類在處理完自己的邏輯后成翩,還是會調(diào)用procee(),傳入this就是為了達到這個目的
            return ((MethodInterceptor) interceptorOrInterceptionAdvice).invoke(this);
        }
    }

類圖

Advisor

Advisor.png

Advice

Advice.png

Joinpoint

Joinpoint.png
最后編輯于
?著作權歸作者所有,轉載或內(nèi)容合作請聯(lián)系作者
  • 序言:七十年代末觅捆,一起剝皮案震驚了整個濱河市,隨后出現(xiàn)的幾起案子麻敌,更是在濱河造成了極大的恐慌栅炒,老刑警劉巖,帶你破解...
    沈念sama閱讀 211,290評論 6 491
  • 序言:濱河連續(xù)發(fā)生了三起死亡事件术羔,死亡現(xiàn)場離奇詭異赢赊,居然都是意外死亡,警方通過查閱死者的電腦和手機级历,發(fā)現(xiàn)死者居然都...
    沈念sama閱讀 90,107評論 2 385
  • 文/潘曉璐 我一進店門,熙熙樓的掌柜王于貴愁眉苦臉地迎上來寥殖,“玉大人玩讳,你說我怎么就攤上這事】盖荩” “怎么了锋边?”我有些...
    開封第一講書人閱讀 156,872評論 0 347
  • 文/不壞的土叔 我叫張陵皱坛,是天一觀的道長编曼。 經(jīng)常有香客問我,道長剩辟,這世上最難降的妖魔是什么掐场? 我笑而不...
    開封第一講書人閱讀 56,415評論 1 283
  • 正文 為了忘掉前任,我火速辦了婚禮贩猎,結果婚禮上熊户,老公的妹妹穿的比我還像新娘。我一直安慰自己吭服,他們只是感情好嚷堡,可當我...
    茶點故事閱讀 65,453評論 6 385
  • 文/花漫 我一把揭開白布。 她就那樣靜靜地躺著,像睡著了一般蝌戒。 火紅的嫁衣襯著肌膚如雪串塑。 梳的紋絲不亂的頭發(fā)上,一...
    開封第一講書人閱讀 49,784評論 1 290
  • 那天北苟,我揣著相機與錄音桩匪,去河邊找鬼。 笑死友鼻,一個胖子當著我的面吹牛傻昙,可吹牛的內(nèi)容都是我干的。 我是一名探鬼主播彩扔,決...
    沈念sama閱讀 38,927評論 3 406
  • 文/蒼蘭香墨 我猛地睜開眼妆档,長吁一口氣:“原來是場噩夢啊……” “哼!你這毒婦竟也來了借杰?” 一聲冷哼從身側響起过吻,我...
    開封第一講書人閱讀 37,691評論 0 266
  • 序言:老撾萬榮一對情侶失蹤,失蹤者是張志新(化名)和其女友劉穎蔗衡,沒想到半個月后纤虽,有當?shù)厝嗽跇淞掷锇l(fā)現(xiàn)了一具尸體,經(jīng)...
    沈念sama閱讀 44,137評論 1 303
  • 正文 獨居荒郊野嶺守林人離奇死亡绞惦,尸身上長有42處帶血的膿包…… 初始之章·張勛 以下內(nèi)容為張勛視角 年9月15日...
    茶點故事閱讀 36,472評論 2 326
  • 正文 我和宋清朗相戀三年逼纸,在試婚紗的時候發(fā)現(xiàn)自己被綠了。 大學時的朋友給我發(fā)了我未婚夫和他白月光在一起吃飯的照片济蝉。...
    茶點故事閱讀 38,622評論 1 340
  • 序言:一個原本活蹦亂跳的男人離奇死亡杰刽,死狀恐怖,靈堂內(nèi)的尸體忽然破棺而出王滤,到底是詐尸還是另有隱情贺嫂,我是刑警寧澤,帶...
    沈念sama閱讀 34,289評論 4 329
  • 正文 年R本政府宣布雁乡,位于F島的核電站第喳,受9級特大地震影響,放射性物質(zhì)發(fā)生泄漏踱稍。R本人自食惡果不足惜曲饱,卻給世界環(huán)境...
    茶點故事閱讀 39,887評論 3 312
  • 文/蒙蒙 一、第九天 我趴在偏房一處隱蔽的房頂上張望珠月。 院中可真熱鬧扩淀,春花似錦、人聲如沸啤挎。這莊子的主人今日做“春日...
    開封第一講書人閱讀 30,741評論 0 21
  • 文/蒼蘭香墨 我抬頭看了看天上的太陽。三九已至胜臊,卻和暖如春氛谜,著一層夾襖步出監(jiān)牢的瞬間,已是汗流浹背区端。 一陣腳步聲響...
    開封第一講書人閱讀 31,977評論 1 265
  • 我被黑心中介騙來泰國打工值漫, 沒想到剛下飛機就差點兒被人妖公主榨干…… 1. 我叫王不留,地道東北人织盼。 一個月前我還...
    沈念sama閱讀 46,316評論 2 360
  • 正文 我出身青樓杨何,卻偏偏與公主長得像,于是被迫代替她去往敵國和親沥邻。 傳聞我的和親對象是個殘疾皇子危虱,可洞房花燭夜當晚...
    茶點故事閱讀 43,490評論 2 348

推薦閱讀更多精彩內(nèi)容

  • spring-AOP(一) 手動代理 spring的設計原理是構造一個個原子功能,然后不斷的通過設計模式在外圍進行...
    立志java閱讀 833評論 1 1
  • 首先唐全,為了讓大家能更有效的理解AOP埃跷,先帶大家過一下AOP中的術語: 切面(Aspect):指關注點模塊化,這個關...
    聯(lián)旺閱讀 353評論 0 0
  • 1邮利、AOP核心概念 1)弥雹、橫切關注點(對哪些方法進行切入) 對哪些方法進行攔截,攔截后怎么處理延届,這些關注點稱之為橫...
    steven_it閱讀 323評論 0 0
  • Spring AOP 機制源碼解讀 導讀 AOP是Spring框架面向切面的編程思想剪勿,AOP采用一種稱為“橫切”的...
    花神子閱讀 418評論 0 3
  • 配置注解版事務 事務管理器 事務管理器的接口是PlatformTransactionManager,其中定義了三個...
    愛撒謊的男孩閱讀 1,281評論 0 3