Spring 源碼之AOP 原理(上)

作為 Spring 體系里的大塊頭校摩,AOP用起來是很爽,但是問你它是怎么實(shí)現(xiàn)的阶淘,你就懵逼衙吩。嘿嘿嘿 ~

還是從 SpringBoot 的啟動(dòng)流程來講起,看看定義切面后的啟動(dòng)流程溪窒。

先看我們的測試?yán)樱?/p>

TestController.java :

TestController.png

LogAspect.java :

LogAspect.png

接下來坤塞,從啟動(dòng)流程看冯勉,TestController 和 LogAspect是怎么創(chuàng)建的。

我們找到 AbstractAutowireCapableBeanFactory#initializeBean() 方法尺锚,如圖:

別問我怎么找到這個(gè)方法的珠闰,我不會(huì)告訴你是打了九九八十一個(gè)斷點(diǎn)調(diào)出來的 =。=

initializeBean.png

進(jìn)入 applyBeanPostProcessorsAfterInitialization(wrappedBean, beanName) 方法瘫辩,看下參數(shù):

applyBeanPostProcessorsAfterInit.png

遍歷到 AnnotationAwareAspectJAutoProxyCreator 這個(gè)processor的時(shí)候伏嗜,debug進(jìn)去,可以看到:


postProcessAfterInit.png

這個(gè) wrapIfNecessary()方法就很關(guān)鍵了伐厌,再點(diǎn)進(jìn)去:

specificInterceptors.png

可以看到這一步里承绸,先是找到所有的通知器(攔截器或者增強(qiáng)器),然后接下來創(chuàng)建proxy:

點(diǎn)擊createProxy()挣轨,進(jìn)入军熏,看核心代碼代碼:

protected Object createProxy(Class<?> beanClass, @Nullable String beanName,
            @Nullable Object[] specificInterceptors, TargetSource targetSource) {

        ……

        ProxyFactory proxyFactory = new ProxyFactory();
        proxyFactory.copyFrom(this);

        ……

        Advisor[] advisors = buildAdvisors(beanName, specificInterceptors);
        proxyFactory.addAdvisors(advisors);
        proxyFactory.setTargetSource(targetSource);
        customizeProxyFactory(proxyFactory);

        proxyFactory.setFrozen(this.freezeProxy);
        if (advisorsPreFiltered()) {
            proxyFactory.setPreFiltered(true);
        }

        ClassLoader classLoader = getProxyClassLoader();
        if (classLoader instanceof SmartClassLoader && classLoader != beanClass.getClassLoader()) {
            classLoader = ((SmartClassLoader) classLoader).getOriginalClassLoader();
        }
        return proxyFactory.getProxy(classLoader);
    }

可以看到,這里都是對(duì) proxyFactory 進(jìn)行設(shè)置的卷扮。點(diǎn)擊 proxyFactory.getProxy(classLoader) 荡澎,一直追到

DefaultAopProxyFactory # createAopProxy( ),這里就是創(chuàng)建代理的核心了:

public AopProxy createAopProxy(AdvisedSupport config) throws AopConfigException {
    // 如果開啟了優(yōu)化晤锹,或者 ProxyTargetClass設(shè)置為true摩幔,或者沒有代理類要實(shí)現(xiàn)的接口
    if (config.isOptimize() || config.isProxyTargetClass() || hasNoUserSuppliedProxyInterfaces(config)) {
        // 如果targetClass(這里是指我們調(diào)試的 TestController.class) 是接口或者代理類,則創(chuàng)建jdk動(dòng)態(tài)代理鞭铆,否則返回 cglib 代理
        if (targetClass.isInterface() || Proxy.isProxyClass(targetClass)) {
            return new JdkDynamicAopProxy(config);
        }
        return new ObjenesisCglibAopProxy(config);
        }
        else {
            return new JdkDynamicAopProxy(config);
        }
    }

createAopProxy() 之后是 getProxy()或衡,以 CglibAopProxy#getProxy() 為例:

public Object getProxy(@Nullable ClassLoader classLoader) {

    try {
        ……
        // 配置 CGLIB 增強(qiáng)器
        Enhancer enhancer = createEnhancer();
        if (classLoader != null) {
            enhancer.setClassLoader(classLoader);
            
            enhancer.setSuperclass(proxySuperClass);        enhancer.setInterfaces(AopProxyUtils.completeProxiedInterfaces(this.advised));
            enhancer.setNamingPolicy(SpringNamingPolicy.INSTANCE);
            enhancer.setStrategy(new ClassLoaderAwareGeneratorStrategy(classLoader));
            // 獲取代理的回調(diào)方法
            Callback[] callbacks = getCallbacks(rootClass);
            Class<?>[] types = new Class<?>[callbacks.length];
            for (int x = 0; x < types.length; x++) {
                types[x] = callbacks[x].getClass();
            }
            enhancer.setCallbackFilter(new ProxyCallbackFilter(
                    this.advised.getConfigurationOnlyCopy(), this.fixedInterceptorMap, this.fixedInterceptorOffset));
            enhancer.setCallbackTypes(types);

            // 生成代理類和實(shí)例
            return createProxyClassAndInstance(enhancer, callbacks);
        }
    }

大概流程就是這樣了〕邓欤總結(jié)一下封断,就是根據(jù)我們的目標(biāo)類,生成jdk動(dòng)態(tài)代理或者cglib代理舶担,后面執(zhí)行到這個(gè)目標(biāo)類(例子中是TestController) 的時(shí)候坡疼,就用生成的代理類去執(zhí)行,而代理類里面有增強(qiáng)器衣陶,也就是我們的橫切面回梧。

下一節(jié)講講cglib 和 jdk 動(dòng)態(tài)代理工作的過程。

最后編輯于
?著作權(quán)歸作者所有,轉(zhuǎn)載或內(nèi)容合作請(qǐng)聯(lián)系作者
  • 序言:七十年代末祖搓,一起剝皮案震驚了整個(gè)濱河市,隨后出現(xiàn)的幾起案子湖苞,更是在濱河造成了極大的恐慌拯欧,老刑警劉巖,帶你破解...
    沈念sama閱讀 222,252評(píng)論 6 516
  • 序言:濱河連續(xù)發(fā)生了三起死亡事件财骨,死亡現(xiàn)場離奇詭異镐作,居然都是意外死亡藏姐,警方通過查閱死者的電腦和手機(jī),發(fā)現(xiàn)死者居然都...
    沈念sama閱讀 94,886評(píng)論 3 399
  • 文/潘曉璐 我一進(jìn)店門该贾,熙熙樓的掌柜王于貴愁眉苦臉地迎上來羔杨,“玉大人,你說我怎么就攤上這事杨蛋《挡模” “怎么了?”我有些...
    開封第一講書人閱讀 168,814評(píng)論 0 361
  • 文/不壞的土叔 我叫張陵逞力,是天一觀的道長曙寡。 經(jīng)常有香客問我,道長寇荧,這世上最難降的妖魔是什么举庶? 我笑而不...
    開封第一講書人閱讀 59,869評(píng)論 1 299
  • 正文 為了忘掉前任,我火速辦了婚禮揩抡,結(jié)果婚禮上户侥,老公的妹妹穿的比我還像新娘。我一直安慰自己峦嗤,他們只是感情好蕊唐,可當(dāng)我...
    茶點(diǎn)故事閱讀 68,888評(píng)論 6 398
  • 文/花漫 我一把揭開白布。 她就那樣靜靜地躺著寻仗,像睡著了一般刃泌。 火紅的嫁衣襯著肌膚如雪。 梳的紋絲不亂的頭發(fā)上署尤,一...
    開封第一講書人閱讀 52,475評(píng)論 1 312
  • 那天耙替,我揣著相機(jī)與錄音,去河邊找鬼曹体。 笑死俗扇,一個(gè)胖子當(dāng)著我的面吹牛,可吹牛的內(nèi)容都是我干的箕别。 我是一名探鬼主播铜幽,決...
    沈念sama閱讀 41,010評(píng)論 3 422
  • 文/蒼蘭香墨 我猛地睜開眼悬而,長吁一口氣:“原來是場噩夢(mèng)啊……” “哼壮池!你這毒婦竟也來了?” 一聲冷哼從身側(cè)響起半沽,我...
    開封第一講書人閱讀 39,924評(píng)論 0 277
  • 序言:老撾萬榮一對(duì)情侶失蹤母截,失蹤者是張志新(化名)和其女友劉穎到忽,沒想到半個(gè)月后,有當(dāng)?shù)厝嗽跇淞掷锇l(fā)現(xiàn)了一具尸體,經(jīng)...
    沈念sama閱讀 46,469評(píng)論 1 319
  • 正文 獨(dú)居荒郊野嶺守林人離奇死亡喘漏,尸身上長有42處帶血的膿包…… 初始之章·張勛 以下內(nèi)容為張勛視角 年9月15日...
    茶點(diǎn)故事閱讀 38,552評(píng)論 3 342
  • 正文 我和宋清朗相戀三年护蝶,在試婚紗的時(shí)候發(fā)現(xiàn)自己被綠了。 大學(xué)時(shí)的朋友給我發(fā)了我未婚夫和他白月光在一起吃飯的照片翩迈。...
    茶點(diǎn)故事閱讀 40,680評(píng)論 1 353
  • 序言:一個(gè)原本活蹦亂跳的男人離奇死亡持灰,死狀恐怖,靈堂內(nèi)的尸體忽然破棺而出负饲,到底是詐尸還是另有隱情堤魁,我是刑警寧澤,帶...
    沈念sama閱讀 36,362評(píng)論 5 351
  • 正文 年R本政府宣布绽族,位于F島的核電站姨涡,受9級(jí)特大地震影響,放射性物質(zhì)發(fā)生泄漏吧慢。R本人自食惡果不足惜涛漂,卻給世界環(huán)境...
    茶點(diǎn)故事閱讀 42,037評(píng)論 3 335
  • 文/蒙蒙 一、第九天 我趴在偏房一處隱蔽的房頂上張望检诗。 院中可真熱鬧匈仗,春花似錦、人聲如沸逢慌。這莊子的主人今日做“春日...
    開封第一講書人閱讀 32,519評(píng)論 0 25
  • 文/蒼蘭香墨 我抬頭看了看天上的太陽攻泼。三九已至火架,卻和暖如春,著一層夾襖步出監(jiān)牢的瞬間忙菠,已是汗流浹背何鸡。 一陣腳步聲響...
    開封第一講書人閱讀 33,621評(píng)論 1 274
  • 我被黑心中介騙來泰國打工, 沒想到剛下飛機(jī)就差點(diǎn)兒被人妖公主榨干…… 1. 我叫王不留牛欢,地道東北人骡男。 一個(gè)月前我還...
    沈念sama閱讀 49,099評(píng)論 3 378
  • 正文 我出身青樓,卻偏偏與公主長得像傍睹,于是被迫代替她去往敵國和親隔盛。 傳聞我的和親對(duì)象是個(gè)殘疾皇子,可洞房花燭夜當(dāng)晚...
    茶點(diǎn)故事閱讀 45,691評(píng)論 2 361

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

  • 一 動(dòng)態(tài)代理 動(dòng)態(tài)代理分為JDK動(dòng)態(tài)代理和CGLIB動(dòng)態(tài)代理 jdk動(dòng)態(tài)代理 被代理類(目標(biāo)類)和代理類必須實(shí)現(xiàn)同...
    今年五年級(jí)閱讀 526評(píng)論 0 6
  • 1拾稳、什么是AOP AOP的核心思想就是“將應(yīng)用程序中的商業(yè)邏輯同對(duì)其提供支持的通用服務(wù)進(jìn)行分離------Avan...
    我從不砍柴閱讀 647評(píng)論 0 0
  • AOP相關(guān)概念 方面(Aspect):一個(gè)關(guān)注點(diǎn)的模塊化,這個(gè)關(guān)注點(diǎn)實(shí)現(xiàn)可能另外橫切多個(gè)對(duì)象。事務(wù)管理是J2EE應(yīng)...
    _LIECO閱讀 10,185評(píng)論 0 0
  • 我們?cè)?Spring AOP原理 (二)AnnotationAwareAspectJAutoProxyCreato...
    就很Nice閱讀 366評(píng)論 0 1
  • 什么是AOP AOP(Aspect-OrientedProgramming俱笛,面向方面編程),可以說是OOP(Obj...
    敲程序的人生閱讀 696評(píng)論 0 0