• <ins id="pjuwb"></ins>
    <blockquote id="pjuwb"><pre id="pjuwb"></pre></blockquote>
    <noscript id="pjuwb"></noscript>
          <sup id="pjuwb"><pre id="pjuwb"></pre></sup>
            <dd id="pjuwb"></dd>
            <abbr id="pjuwb"></abbr>
            天空之城
            new,think,program,happy to live
            posts - 39,comments - 39,trackbacks - 0
            NET委托:一個(gè)C#睡前故事
            英文版原作者:Chris Sells(www.sellsbrothers.com
            翻譯:袁曉輝(www.farproc.comhttp://blog.csdn.net/uoyevoli

            緊耦合
            從前,在南方一塊奇異的土地上,有個(gè)工人名叫彼得,他非常勤奮,對他的老板總是百依百順。但是他的老板是個(gè)吝嗇的人,從不信任別人,堅(jiān)決要求隨時(shí)知道彼得的工作進(jìn)度,以防止他偷懶。但是彼得又不想讓老板呆在他的辦公室里站在背后盯著他,于是就對老板做出承諾:無論何時(shí),只要我的工作取得了一點(diǎn)進(jìn)展我都會(huì)及時(shí)讓你知道。彼得通過周期性地使用“帶類型的引用”(原文為:“typed reference” 也就是delegate??)“回調(diào)”他的老板來實(shí)現(xiàn)他的承諾,如下:
            class Worker {
            ??? public void Advise(Boss boss) { _boss = boss; }
            ??? public void DoWork() {
            ??????? Console.WriteLine(“工作: 工作開始”);
            ??????? if( _boss != null ) _boss.WorkStarted();

            ??????? Console.WriteLine(“工作: 工作進(jìn)行中”);
            ??????? if( _boss != null ) _boss.WorkProgressing();

            ??????? Console.WriteLine("“工作: 工作完成”");
            ??????? if( _boss != null ) {
            ??????????? int grade = _boss.WorkCompleted();
            ??????????? Console.WriteLine(“工人的工作得分=” + grade);
            ??? }
            }
            private Boss _boss;
            }

            class Boss {
            ??? public void WorkStarted() { /* 老板不關(guān)心。 */ }
            ??? public void WorkProgressing() { /*老板不關(guān)心。 */ }
            ??? public int WorkCompleted() {
            ??????? Console.WriteLine(“時(shí)間差不多!”);
            ??????? return 2; /* 總分為10 */
            ??? }
            }

            class Universe {
            ??? static void Main() {
            ??????? Worker peter = new Worker();
            ??????? Boss boss = new Boss();
            ??????? peter.Advise(boss);
            ??????? peter.DoWork();

            ??????? Console.WriteLine(“Main: 工人工作完成”);
            ??????? Console.ReadLine();
            ??? }
            }


            接口

            現(xiàn)在,彼得成了一個(gè)特殊的人,他不但能容忍吝嗇的老板,而且和他周圍的宇宙也有了密切的聯(lián)系,以至于他認(rèn)為宇宙對他的工作進(jìn)度也感興趣。不幸的是,他必須也給宇宙添加一個(gè)特殊的回調(diào)函數(shù)Advise來實(shí)現(xiàn)同時(shí)向他老板和宇宙報(bào)告工作進(jìn)度。彼得想要把潛在的通知的列表和這些通知的實(shí)現(xiàn)方法分離開來,于是他決定把方法分離為一個(gè)接口:

            interface IWorkerEvents {
            ??? void WorkStarted();
            ??? void WorkProgressing();
            ??? int WorkCompleted();
            }

            class Worker {
            ??? public void Advise(IWorkerEvents events) { _events = events; }
            ??? public void DoWork() {
            ??????? Console.WriteLine(“工作: 工作開始”);
            ??????? if( _events != null ) _events.WorkStarted();

            ??????? Console.WriteLine(“工作: 工作進(jìn)行中”);
            ??????? if(_events != null ) _events.WorkProgressing();

            ??????? Console.WriteLine("“工作: 工作完成”");
            ??????? if(_events != null ) {
            ??????????? int grade = _events.WorkCompleted();

            ??????????? Console.WriteLine(“工人的工作得分=” + grade);
            ??????????? }
            ??? }
            ??? private IWorkerEvents _events;
            }

            class Boss : IWorkerEvents {
            ??? public void WorkStarted() { /* 老板不關(guān)心。 */ }
            ??? public void WorkProgressing() { /* 老板不關(guān)心。 */ }
            ??? public int WorkCompleted() {
            ??????? Console.WriteLine(“時(shí)間差不多!”);
            ??????? return 3; /* 總分為10 */
            ??? }
            }


            委托

            不幸的是,每當(dāng)彼得忙于通過接口的實(shí)現(xiàn)和老板交流時(shí),就沒有機(jī)會(huì)及時(shí)通知宇宙了。至少他應(yīng)該忽略身在遠(yuǎn)方的老板的引用,好讓其他實(shí)現(xiàn)了IWorkerEvents的對象得到他的工作報(bào)告。(”At least he'd abstracted the reference of his boss far away from him so that others who implemented the IWorkerEvents interface could be notified of his work progress” 原話如此,不理解到底是什么意思 )

            他的老板還是抱怨得很厲害。“彼得!”他老板吼道,“你為什么在工作一開始和工作進(jìn)行中都來煩我?!我不關(guān)心這些事件。你不但強(qiáng)迫我實(shí)現(xiàn)了這些方法,而且還在浪費(fèi)我寶貴的工作時(shí)間來處理你的事件,特別是當(dāng)我外出的時(shí)候更是如此!你能不能不再來煩我?”

            于是,彼得意識(shí)到接口雖然在很多情況都很有用,但是當(dāng)用作事件時(shí),“粒度”不夠好。他希望能夠僅在別人想要時(shí)才通知他們,于是他決定把接口的方法分離為單獨(dú)的委托,每個(gè)委托都像一個(gè)小的接口方法:

            delegate void WorkStarted();
            delegate void WorkProgressing();
            delegate int WorkCompleted();

            class Worker {
            ??? public void DoWork() {
            ??????? Console.WriteLine(“工作: 工作開始”);
            ??????? if( started != null ) started();

            ??????? Console.WriteLine(“工作: 工作進(jìn)行中”);
            ??????? if( progressing != null ) progressing();

            ??????? Console.WriteLine("“工作: 工作完成”");
            ??????? if( completed != null ) {
            ??????????? int grade = completed();
            ??????????? Console.WriteLine(“工人的工作得分=” + grade);
            ??????? }
            ??? }
            ??? public WorkStarted started;
            ??? public WorkProgressing progressing;
            ??? public WorkCompleted completed;
            }

            class Boss {
            ??? public int WorkCompleted() {
            ??? Console.WriteLine("Better...");
            ??? return 4; /* 總分為10 */
            }
            }

            class Universe {
            ??? static void Main() {
            ??????? Worker peter = new Worker();
            ??????? Boss boss = new Boss();
            ??????? peter.completed = new WorkCompleted(boss.WorkCompleted);
            ??????? peter.DoWork();

            ??????? Console.WriteLine(“Main: 工人工作完成”);
            ??????? Console.ReadLine();
            ??? }
            }


            靜態(tài)監(jiān)聽者

            這樣,彼得不會(huì)再拿他老板不想要的事件來煩他老板了,但是他還沒有把宇宙放到他的監(jiān)聽者列表中。因?yàn)橛钪媸莻€(gè)包涵一切的實(shí)體,看來不適合使用實(shí)例方法的委托(想像一下,實(shí)例化一個(gè)“宇宙”要花費(fèi)多少資源…..),于是彼得就需要能夠?qū)o態(tài)委托進(jìn)行掛鉤,委托對這一點(diǎn)支持得很好:

            class Universe {
            ??? static void WorkerStartedWork() {
            ??????? Console.WriteLine("Universe notices worker starting work");
            ??? }

            ??? static int WorkerCompletedWork() {
            ??????? Console.WriteLine("Universe pleased with worker's work");
            ??????? return 7;
            ??? }

            ??? static void Main() {
            ??????? Worker peter = new Worker();
            ??????? Boss boss = new Boss();
            ??????? peter.completed = new WorkCompleted(boss.WorkCompleted);
            ??????? peter.started = new WorkStarted(Universe.WorkerStartedWork);
            ??????? peter.completed = new WorkCompleted(Universe.WorkerCompletedWork);
            ??????? peter.DoWork();

            ??????? Console.WriteLine(“Main: 工人工作完成”);
            ??????? Console.ReadLine();
            ??? }
            }


            事件

            不幸的是,宇宙太忙了,也不習(xí)慣時(shí)刻關(guān)注它里面的個(gè)體,它可以用自己的委托替換了彼得老板的委托。這是把彼得的Worker類的的委托字段做成public的一個(gè)無意識(shí)的副作用。同樣,如果彼得的老板不耐煩了,也可以決定自己來激發(fā)彼得的委托(真是一個(gè)粗魯?shù)睦习澹?/p>

            // Peter's boss taking matters into his own hands
            if( peter.completed != null ) peter.completed();
            彼得不想讓這些事發(fā)生,他意識(shí)到需要給每個(gè)委托提供“注冊”和“反注冊”功能,這樣監(jiān)聽者就可以自己添加和移除委托,但同時(shí)又不能清空整個(gè)列表也不能隨意激發(fā)彼得的事件了。彼得并沒有來自己實(shí)現(xiàn)這些功能,相反,他使用了event關(guān)鍵字讓C#編譯器為他構(gòu)建這些方法:

            class Worker {
            ...
            ??? public event WorkStarted started;
            ??? public event WorkProgressing progressing;
            ??? public event WorkCompleted completed;
            }


            彼得知道event關(guān)鍵字在委托的外邊包裝了一個(gè)property,僅讓C#客戶通過+= 和 -=操作符來添加和移除,強(qiáng)迫他的老板和宇宙正確地使用事件。

            static void Main() {
            ??? Worker peter = new Worker();
            ??? Boss boss = new Boss();
            ??? peter.completed += new WorkCompleted(boss.WorkCompleted);
            ??? peter.started += new WorkStarted(Universe.WorkerStartedWork);
            ??? peter.completed += new WorkCompleted(Universe.WorkerCompletedWork);
            ??? peter.DoWork();

            ??? Console.WriteLine(“Main: 工人工作完成”);
            ??? Console.ReadLine();
            }


            “收獲”所有結(jié)果

            到這時(shí),彼得終于可以送一口氣了,他成功地滿足了所有監(jiān)聽者的需求,同時(shí)避免了與特定實(shí)現(xiàn)的緊耦合。但是他注意到他的老板和宇宙都為它的工作打了分,但是他僅僅接收了一個(gè)分?jǐn)?shù)。面對多個(gè)監(jiān)聽者,他想要“收獲”所有的結(jié)果,于是他深入到代理里面,輪詢監(jiān)聽者列表,手工一個(gè)個(gè)調(diào)用:

            public void DoWork() {
            ??? ...
            ??? Console.WriteLine("“工作: 工作完成”");
            ??? if( completed != null ) {
            ??????? foreach( WorkCompleted wc in completed.GetInvocationList() ) {
            ??????????? int grade = wc();
            ??????????? Console.WriteLine(“工人的工作得分=” + grade);
            ??????? }
            ??? }
            }


            異步通知:激發(fā) & 忘掉

            同時(shí),他的老板和宇宙還要忙于處理其他事情,也就是說他們給彼得打分所花費(fèi)的事件變得非常長:

            class Boss {
            ??? public int WorkCompleted() {
            ??????? System.Threading.Thread.Sleep(3000);
            ??????? Console.WriteLine("Better..."); return 6; /* 總分為10 */
            ??? }
            }

            class Universe {
            ??? static int WorkerCompletedWork() {
            ??????? System.Threading.Thread.Sleep(4000);
            ??????? Console.WriteLine("Universe is pleased with worker's work");
            ??????? return 7;
            ??? }
            ??? ...
            }
            很不幸,彼得每次通知一個(gè)監(jiān)聽者后必須等待它給自己打分,現(xiàn)在這些通知花費(fèi)了他太多的工作事件。于是他決定忘掉分?jǐn)?shù),僅僅異步激發(fā)事件:

            public void DoWork() {
            ??? ...
            ??? Console.WriteLine("“工作: 工作完成”");
            ??? if( completed != null ) {
            ??????? foreach( WorkCompleted wc in completed.GetInvocationList() )
            ??????? {
            ??????????? wc.BeginInvoke(null, null);
            ??????? }
            ??? }
            }


            異步通知:輪詢

            這使得彼得可以通知他的監(jiān)聽者,然后立即返回工作,讓進(jìn)程的線程池來調(diào)用這些代理。隨著時(shí)間的過去,彼得發(fā)現(xiàn)他丟失了他工作的反饋,他知道聽取別人的贊揚(yáng)和努力工作一樣重要,于是他異步激發(fā)事件,但是周期性地輪詢,取得可用的分?jǐn)?shù)。

            public void DoWork() {
            ??? ...
            ??? Console.WriteLine("“工作: 工作完成”");
            ??? if( completed != null ) {
            ??????? foreach( WorkCompleted wc in completed.GetInvocationList() ) {
            ??????????? IAsyncResult res = wc.BeginInvoke(null, null);
            ??????????? while( !res.IsCompleted ) System.Threading.Thread.Sleep(1);
            ??????????? int grade = wc.EndInvoke(res);
            ??????????? Console.WriteLine(“工人的工作得分=” + grade);
            ??????? }
            ??? }
            }


            異步通知:委托

            不幸地,彼得有回到了一開始就想避免的情況中來,比如,老板站在背后盯著他工作。于是,他決定使用自己的委托作為他調(diào)用的異步委托完成的通知,讓他自己立即回到工作,但是仍可以在別人給他的工作打分后得到通知:

            ??? public void DoWork() {
            ??????? ...
            ??????? Console.WriteLine("“工作: 工作完成”");
            ??????? if( completed != null ) {
            ??????????? foreach( WorkCompleted wc in completed.GetInvocationList() ) {
            ??????????????? wc.BeginInvoke(new AsyncCallback(WorkGraded), wc);
            ??????????? }
            ??????? }
            ??? }

            ??? private void WorkGraded(IAsyncResult res) {
            ??????? WorkCompleted wc = (WorkCompleted)res.AsyncState;
            ??????? int grade = wc.EndInvoke(res);
            ??????? Console.WriteLine(“工人的工作得分=” + grade);
            ??? }


            宇宙中的幸福

            彼得、他的老板和宇宙最終都滿足了。彼得的老板和宇宙可以收到他們感興趣的事件通知,減少了實(shí)現(xiàn)的負(fù)擔(dān)和非必需的往返“差旅費(fèi)”。彼得可以通知他們,而不管他們要花多長時(shí)間來從目的方法中返回,同時(shí)又可以異步地得到他的結(jié)果。彼得知道,這并不*十分*簡單,因?yàn)楫?dāng)他異步激發(fā)事件時(shí),方法要在另外一個(gè)線程中執(zhí)行,彼得的目的方法完成的通知也是一樣的道理。但是,邁克和彼得是好朋友,他很熟悉線程的事情,可以在這個(gè)領(lǐng)域提供指導(dǎo)。

            ?

            他們永遠(yuǎn)幸福地生活下去……<完>

            posted on 2006-04-08 16:46 太極虎~宏 閱讀(103) 評(píng)論(0)  編輯 收藏 引用

            只有注冊用戶登錄后才能發(fā)表評(píng)論。
            網(wǎng)站導(dǎo)航: 博客園   IT新聞   BlogJava   博問   Chat2DB   管理


            国产精品久久久久乳精品爆| 久久久WWW成人免费毛片| 久久精品日日躁夜夜躁欧美| 久久精品桃花综合| 久久精品国产亚洲AV嫖农村妇女| 久久精品国产亚洲av麻豆色欲| 青青青青久久精品国产 | 久久综合色区| 色综合久久无码中文字幕| 久久综合久久综合九色| 久久AAAA片一区二区| 亚洲AV无码1区2区久久| 国产精品欧美亚洲韩国日本久久| 久久无码高潮喷水| 一本色道久久88加勒比—综合| 久久人人爽人人爽人人片AV东京热 | 中文字幕无码久久人妻| 久久天堂AV综合合色蜜桃网 | 久久99国产精品久久99| 国产午夜精品久久久久九九电影 | 99久久婷婷国产综合亚洲| 久久久综合香蕉尹人综合网| 国产精品美女久久久| 无码人妻精品一区二区三区久久 | 午夜精品久久久久9999高清| 久久精品国产99国产电影网| 综合人妻久久一区二区精品| 久久中文精品无码中文字幕| 国产精品美女久久久久av爽| 久久国产乱子精品免费女| 亚洲国产精品无码成人片久久| 亚洲国产精品无码久久九九| 久久精品国产一区二区三区不卡| 久久综合综合久久狠狠狠97色88| 亚洲色大成网站www久久九| 久久天天躁夜夜躁狠狠躁2022 | 欧美国产成人久久精品| 青青草国产97免久久费观看| 久久99精品久久久久久9蜜桃| 99久久中文字幕| 四虎国产精品免费久久久|