posts - 15, comments - 20, trackbacks - 0, articles - 6
            BlogJava :: 首頁 :: 新隨筆 :: 聯(lián)系 :: 聚合  :: 管理

          aop筆記

          Posted on 2009-09-01 10:58 spinage 閱讀(138) 評論(0)  編輯  收藏
          1.我所知道的aop 
              初看aop,上來就是一大堆術(shù)語,而且還有個拉風的名字,面向切面編程,都說是OOP的一種有益補充等等。一下子讓你不知所措,心想著:怪不得很多人都和我說aop多難多難。當我看進去以后,我才發(fā)現(xiàn):它就是一些java基礎(chǔ)上的樸實無華的應(yīng)用,包括ioc,包括許許多多這樣的名詞,都是萬變不離其宗而已。
          2.為什么用aop 
              1就是為了方便,看一個國外很有名的大師說,編程的人都是“懶人”,因為他把自己做的事情都讓程序做了。用了aop能讓你少寫很多代碼,這點就夠充分了吧
              2就是為了更清晰的邏輯,可以讓你的業(yè)務(wù)邏輯去關(guān)注自己本身的業(yè)務(wù),而不去想一些其他的事情,這些其他的事情包括:安全,事物,日志等。 
          3.那些aop的術(shù)語 
              初看這么多術(shù)語,一下子都不好接受,慢慢來,很快就會搞懂。
              1.通知(Advice)
              就是你想要的功能,也就是上面說的 安全,事物,日志等。你給先定義好把,然后在想用的地方用一下。
              2.連接點(JoinPoint)
              這個更好解釋了,就是spring允許你使用通知的地方,那可真就多了,基本每個方法的前,后(兩者都有也行),或拋出異常時都可以是連接點,spring只支持方法連接點.其他如aspectJ還可以讓你在構(gòu)造器或?qū)傩宰⑷霑r都行,不過那不是咱關(guān)注的,只要記住,和方法有關(guān)的前前后后(拋出異常),都是連接點。
              3.切入點(Pointcut)
              上面說的連接點的基礎(chǔ)上,來定義切入點,你的一個類里,有15個方法,那就有幾十個連接點了對把,但是你并不想在所有方法附近都使用通知(使用叫織入,以后再說),你只想讓其中的幾個,在調(diào)用這幾個方法之前,之后或者拋出異常時干點什么,那么就用切點來定義這幾個方法,讓切點來篩選連接點,選中那幾個你想要的方法。
              4.切面(Aspect)
              切面是通知和切入點的結(jié)合。現(xiàn)在發(fā)現(xiàn)了吧,沒連接點什么事情,連接點就是為了讓你好理解切點,搞出來的,明白這個概念就行了。通知說明了干什么和什么時候干(什么時候通過方法名中的before,after,around等就能知道),而切入點說明了在哪干(指定到底是哪個方法),這就是一個完整的切面定義。
              5.引入(introduction)
              允許我們向現(xiàn)有的類添加新方法屬性。這不就是把切面(也就是新方法屬性:通知定義的)用到目標類中嗎
              6.目標(target)
              引入中所提到的目標類,也就是要被通知的對象,也就是真正的業(yè)務(wù)邏輯,他可以在毫不知情的情況下,被咱們織入切面。而自己專注于業(yè)務(wù)本身的邏輯。
              7.代理(proxy)
              怎么實現(xiàn)整套aop機制的,都是通過代理,這個一會給細說。
              8.織入(weaving)
              把切面應(yīng)用到目標對象來創(chuàng)建新的代理對象的過程。有3種方式,spring采用的是運行時,為什么是運行時,后面解釋。
          關(guān)鍵就是:切點定義了哪些連接點會得到通知 

           4.我所理解的aop原理 
              spring用代理類包裹切面,把他們織入到Spring管理的bean中。也就是說代理類偽裝成目標類,它會截取對目標類中方法的調(diào)用,讓調(diào)用者對目標類的調(diào)用都先變成調(diào)用偽裝類,偽裝類中就先執(zhí)行了切面,再把調(diào)用轉(zhuǎn)發(fā)給真正的目標bean。
              現(xiàn)在可以自己想一想,怎么搞出來這個偽裝類,才不會被調(diào)用者發(fā)現(xiàn)(過JVM的檢查,JAVA是強類型檢查,哪里都要檢查類型)。
              1.實現(xiàn)和目標類相同的接口,我也實現(xiàn)和你一樣的接口,反正上層都是接口級別的調(diào)用,這樣我就偽裝成了和目標類一樣的類(實現(xiàn)了同一接口,咱是兄弟了),也就逃過了類型檢查,到j(luò)ava運行期的時候,利用多態(tài)的后期綁定(所以spring采用運行時),偽裝類(代理類)就變成了接口的真正實現(xiàn),而他里面包裹了真實的那個目標類,最后實現(xiàn)具體功能的還是目標類,只不過偽裝類在之前干了點事情(寫日志,安全檢查,事物等)。
              這就好比,一個人讓你辦件事,每次這個時候,你弟弟就會先出來,當然他分不出來了,以為是你,你這個弟弟雖然辦不了這事,但是他知道你能辦,所以就答應(yīng)下來了,并且收了點禮物(寫日志),收完禮物了,給把事給人家辦了啊,所以你弟弟又找你這個哥哥來了,最后把這是辦了的還是你自己。但是你自己并不知道你弟弟已經(jīng)收禮物了,你只是專心把這件事情做好。
             
              順著這個思路想,要是本身這個類就沒實現(xiàn)一個接口呢,你怎么偽裝我,我就壓根沒有機會讓你搞出這個雙胞胎的弟弟,那么就用第2種代理方式,創(chuàng)建一個目標類的子類,生個兒子,讓兒子偽裝我

              2.生成子類調(diào)用,這次用子類來做為偽裝類,當然這樣也能逃過JVM的強類型檢查,我繼承的嗎,當然查不出來了,子類重寫了目標類的所有方法,當然在這些重寫的方法中,不僅實現(xiàn)了目標類的功能,還在這些功能之前,實現(xiàn)了一些其他的(寫日志,安全檢查,事物等)。
             這次的對比就是,兒子先從爸爸那把本事都學會了,所有人都找兒子辦事情,但是兒子每次辦和爸爸同樣的事之前,都要收點小禮物(寫日志),然后才去辦真正的事。當然爸爸是不知道兒子這么干的了。 這里就有件事情要說,某些本事是爸爸獨有的(final的),兒子學不了,學不了就辦不了這件事,辦不了這個事情,自然就不能收人家禮了。

             前一種兄弟模式,spring會使用JDK的java.lang.reflect.Proxy類,它允許Spring動態(tài)生成一個新類來實現(xiàn)必要的接口,織入通知,并且把對這些接口的任何調(diào)用都轉(zhuǎn)發(fā)到目標類。

              后一種父子模式,spring使用CGLIB庫生成目標類的一個子類,在創(chuàng)建這個子類的時候,spring織入通知,并且把對這個子類的調(diào)用委托到目標類。
              相比之下,還是兄弟模式好些,他能更好的實現(xiàn)松耦合,尤其在今天都高喊著面向接口編程的情況下,父子模式只是在沒有實現(xiàn)接口的時候,也能織入通知,應(yīng)當做一種例外。
          初學aop對aop術(shù)語和原理上的一些總結(jié)

          只有注冊用戶登錄后才能發(fā)表評論。


          網(wǎng)站導(dǎo)航:
           
          主站蜘蛛池模板: 桐柏县| 大埔县| 大化| 牙克石市| 滁州市| 疏勒县| 黄山市| 仁化县| 安丘市| 鄂温| 屏南县| 巴东县| 罗江县| 迭部县| 玉田县| 柯坪县| 塔城市| 紫阳县| 潞西市| 德格县| 兴宁市| 金山区| 崇仁县| 安义县| 商城县| 四子王旗| 合肥市| 肇东市| 彩票| 青神县| 南京市| 图木舒克市| 乡宁县| 沽源县| 上饶市| 深水埗区| 武宁县| 吉首市| 门源| 屯留县| 高碑店市|