在.Net中實現(xiàn)自己的AOP

  • RealProxy基本代理類

    RealProxy類提供代理的基本功能。這個類中有一個GetTransparentProxy方法馒稍,此方法返回當前代理實例的透明代理礁哄。這是我們AOP實現(xiàn)的主要依賴。

    新建一個代理類MyProxy繼承RealProxy铜涉,IIntercept是我們自己實現(xiàn)的攔截器接口智玻,內(nèi)部只有一個Do方法

        private object _target; //當前代理實例
        private List<IIntercept> _intercepts; //攔截器
        MyProxy(object target, Type type,params IIntercept[] intercepts) : base(type)
        {
            _target = target;
            _intercepts = intercepts!=null? intercepts.ToList():null;
        }
    

    RealProxyInvoke方法會執(zhí)行當前代理實例中方法,所以我們可以重寫Invoke來實現(xiàn)AOP

    public override IMessage Invoke(IMessage msg)
    {
        var ctr = msg as IConstructionCallMessage;
        if (ctr != null)//執(zhí)行構(gòu)造函數(shù)
        {
            Console.WriteLine("Construction");
            RealProxy _proxy = RemotingServices.GetRealProxy(this._target);
            _proxy.InitializeServerObject(ctr);
            MarshalByRefObject tp = (MarshalByRefObject)this.GetTransparentProxy();
            return EnterpriseServicesHelper.CreateConstructionReturnMessage(ctr, tp);
        }
        //執(zhí)行當前代理實例方法
        if(_intercepts!=null)
        {
            foreach (var _intercept in _intercepts)
            {
                _intercept.Do();
            }
        }
        var call = msg as IMethodCallMessage;
        Console.WriteLine(string.Format("proxy method:{0}", call.MethodName));
        var result = call.MethodBase.Invoke(this._target,call.Args);
        return new ReturnMessage(result,new object[0],0,null,call);
    }
    

    封裝代理類的實現(xiàn)芙代,注意:RealProxy構(gòu)造函數(shù)要求目標代理類必須必須從 MarshalByRef類型派生吊奢,所以我們需要繼承 MarshalByRefObject類或者 ContextBoundObject類**

    public static class ActivatorContainer
    {
        public static T Create<T>(params IIntercept[] intercepts)
        {
             //構(gòu)造函數(shù)不會被代理
            var result= Activator.CreateInstance(typeof(T));
            var myProxy = new MyProxy(result, typeof(T), intercepts);
            return (T)myProxy.GetTransparentProxy();
        }
    }
    
    var log = new LogIntercept();
    var time = new TimeIntercept();
    var hance = ActivatorContainer.Create<Person>(log,time);
    hance.Say("hello Proxy");
    

    運行結(jié)果:

    Alt text

    熟悉Castle的動態(tài)代理的同學可能會發(fā)現(xiàn),我們自己實現(xiàn)的就像Castle的簡易版纹烹,基于上面的代碼我們也可以實現(xiàn)一個完善的動態(tài)代理

  • ProxyAttribute特性

    ProxyAttribute特性標志指示對象類型需要自定義代理页滚。使用特性的相當于代替了上面的ActivatorContainer靜態(tài)類來自動實現(xiàn)類的代理

    [AttributeUsage(AttributeTargets.Class)]
    public class MyProxyAttribute : ProxyAttribute
    {
        private List<Type> _types; //攔截器
        //注:特性參數(shù)限制于bool, byte, char, double, float, int, long, short, string, System.Type, object, enum。
        public MyProxyAttribute(params Type[] types)
        {
            _types = types?.ToList();
        }
        public override MarshalByRefObject CreateInstance(Type serverType)
        {
            System.Console.WriteLine("Start铺呵!");
            var target= base.CreateInstance(serverType);
            List<IIntercept> intercepts = null;
            if (_types!=null)
            {
                intercepts = new List<IIntercept>();
                intercepts.AddRange(_types.Select(s =>
                {
                    return (IIntercept)Activator.CreateInstance(s);
                }));
            } 
            var myProxy = new MyProxy(target, serverType, intercepts?.ToArray());
            return (MarshalByRefObject)myProxy.GetTransparentProxy();
        }
    }
    

    此時裹驰,我們只要在需要代理的類上加上MyProxyAttribute標志,代理類需要繼承 ContextBoundObject類片挂,而不能是 MarshalByRefObject幻林,然后直接new相關(guān)類就行了。此種實現(xiàn)方式構(gòu)造函數(shù)也會被代理音念,可以通過特性參數(shù)注入攔截器類型來實現(xiàn)相關(guān)攔截器沪饺,相對于直接new攔截器來說更為方便,但不夠靈活症昏。但有一個好處就是如果沒有繼承ContextBoundObject類随闽,目標類不會被代理父丰,也不會不會報錯桃笙。而上面的實現(xiàn)方式如果沒有繼承MarshalByRefObject類啃沪,運行則會報錯。

    var hance = new Person();
    hance.Say("hello");
    

    運行結(jié)果:

    Alt text
  • ContextAttribute特性

    上面兩種AOP都依賴于RealProxy這個基本代理類來實現(xiàn)的,攔截器的攔截方式都比較僵硬扎即,如果需要具體某個攔截器是否可以攔截到某個方法的話,需要在Invoke方法內(nèi)寫大量代碼用于判斷谦秧,這樣則會造成Invoke方法內(nèi)的代碼膨脹伯复,最后難以維護「澹或者參考Castle的攔截選擇器和代理鉤子芋齿,通過外部傳入配置參數(shù)來進行攔截判斷。除此之外成翩,還有一種方式就是通過ContextAttribute特性和IContributeObjectSink接口來實現(xiàn)觅捆。

最后編輯于
?著作權(quán)歸作者所有,轉(zhuǎn)載或內(nèi)容合作請聯(lián)系作者
  • 序言:七十年代末,一起剝皮案震驚了整個濱河市麻敌,隨后出現(xiàn)的幾起案子栅炒,更是在濱河造成了極大的恐慌,老刑警劉巖,帶你破解...
    沈念sama閱讀 222,729評論 6 517
  • 序言:濱河連續(xù)發(fā)生了三起死亡事件赢赊,死亡現(xiàn)場離奇詭異乙漓,居然都是意外死亡,警方通過查閱死者的電腦和手機释移,發(fā)現(xiàn)死者居然都...
    沈念sama閱讀 95,226評論 3 399
  • 文/潘曉璐 我一進店門叭披,熙熙樓的掌柜王于貴愁眉苦臉地迎上來,“玉大人玩讳,你說我怎么就攤上這事趋观。” “怎么了锋边?”我有些...
    開封第一講書人閱讀 169,461評論 0 362
  • 文/不壞的土叔 我叫張陵皱坛,是天一觀的道長。 經(jīng)常有香客問我豆巨,道長剩辟,這世上最難降的妖魔是什么? 我笑而不...
    開封第一講書人閱讀 60,135評論 1 300
  • 正文 為了忘掉前任往扔,我火速辦了婚禮贩猎,結(jié)果婚禮上,老公的妹妹穿的比我還像新娘萍膛。我一直安慰自己吭服,他們只是感情好,可當我...
    茶點故事閱讀 69,130評論 6 398
  • 文/花漫 我一把揭開白布蝗罗。 她就那樣靜靜地躺著艇棕,像睡著了一般。 火紅的嫁衣襯著肌膚如雪串塑。 梳的紋絲不亂的頭發(fā)上沼琉,一...
    開封第一講書人閱讀 52,736評論 1 312
  • 那天,我揣著相機與錄音桩匪,去河邊找鬼打瘪。 笑死,一個胖子當著我的面吹牛傻昙,可吹牛的內(nèi)容都是我干的闺骚。 我是一名探鬼主播,決...
    沈念sama閱讀 41,179評論 3 422
  • 文/蒼蘭香墨 我猛地睜開眼妆档,長吁一口氣:“原來是場噩夢啊……” “哼僻爽!你這毒婦竟也來了?” 一聲冷哼從身側(cè)響起过吻,我...
    開封第一講書人閱讀 40,124評論 0 277
  • 序言:老撾萬榮一對情侶失蹤进泼,失蹤者是張志新(化名)和其女友劉穎蔗衡,沒想到半個月后,有當?shù)厝嗽跇淞掷锇l(fā)現(xiàn)了一具尸體乳绕,經(jīng)...
    沈念sama閱讀 46,657評論 1 320
  • 正文 獨居荒郊野嶺守林人離奇死亡绞惦,尸身上長有42處帶血的膿包…… 初始之章·張勛 以下內(nèi)容為張勛視角 年9月15日...
    茶點故事閱讀 38,723評論 3 342
  • 正文 我和宋清朗相戀三年,在試婚紗的時候發(fā)現(xiàn)自己被綠了洋措。 大學時的朋友給我發(fā)了我未婚夫和他白月光在一起吃飯的照片济蝉。...
    茶點故事閱讀 40,872評論 1 353
  • 序言:一個原本活蹦亂跳的男人離奇死亡,死狀恐怖菠发,靈堂內(nèi)的尸體忽然破棺而出王滤,到底是詐尸還是另有隱情,我是刑警寧澤滓鸠,帶...
    沈念sama閱讀 36,533評論 5 351
  • 正文 年R本政府宣布雁乡,位于F島的核電站,受9級特大地震影響糜俗,放射性物質(zhì)發(fā)生泄漏踱稍。R本人自食惡果不足惜,卻給世界環(huán)境...
    茶點故事閱讀 42,213評論 3 336
  • 文/蒙蒙 一悠抹、第九天 我趴在偏房一處隱蔽的房頂上張望珠月。 院中可真熱鬧,春花似錦楔敌、人聲如沸啤挎。這莊子的主人今日做“春日...
    開封第一講書人閱讀 32,700評論 0 25
  • 文/蒼蘭香墨 我抬頭看了看天上的太陽庆聘。三九已至,卻和暖如春氛谜,著一層夾襖步出監(jiān)牢的瞬間掏觉,已是汗流浹背。 一陣腳步聲響...
    開封第一講書人閱讀 33,819評論 1 274
  • 我被黑心中介騙來泰國打工值漫, 沒想到剛下飛機就差點兒被人妖公主榨干…… 1. 我叫王不留,地道東北人织盼。 一個月前我還...
    沈念sama閱讀 49,304評論 3 379
  • 正文 我出身青樓杨何,卻偏偏與公主長得像,于是被迫代替她去往敵國和親沥邻。 傳聞我的和親對象是個殘疾皇子危虱,可洞房花燭夜當晚...
    茶點故事閱讀 45,876評論 2 361

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

  • Spring Cloud為開發(fā)人員提供了快速構(gòu)建分布式系統(tǒng)中一些常見模式的工具(例如配置管理,服務(wù)發(fā)現(xiàn)唐全,斷路器埃跷,智...
    卡卡羅2017閱讀 134,716評論 18 139
  • 對象的創(chuàng)建與銷毀 Item 1: 使用static工廠方法蕊玷,而不是構(gòu)造函數(shù)創(chuàng)建對象:僅僅是創(chuàng)建對象的方法,并非Fa...
    孫小磊閱讀 1,996評論 0 3
  • 1. Java基礎(chǔ)部分 基礎(chǔ)部分的順序:基本語法弥雹,類相關(guān)的語法垃帅,內(nèi)部類的語法,繼承相關(guān)的語法剪勿,異常的語法贸诚,線程的語...
    子非魚_t_閱讀 31,665評論 18 399
  • 轉(zhuǎn)至元數(shù)據(jù)結(jié)尾創(chuàng)建: 董瀟偉,最新修改于: 十二月 23, 2016 轉(zhuǎn)至元數(shù)據(jù)起始第一章:isa和Class一....
    40c0490e5268閱讀 1,732評論 0 9
  • 今天中午和同事聊天厕吉,發(fā)現(xiàn)自己這一年多的開發(fā)做的迷迷糊糊酱固,寫過的東西再提起來還是一臉懵逼,于是開啟寫筆記之路头朱。記錄...
    寧寧寧寧寧曉曼閱讀 733評論 0 0