青青草原综合久久大伊人导航_色综合久久天天综合_日日噜噜夜夜狠狠久久丁香五月_热久久这里只有精品

用C++模擬C#的event機(jī)制

作為一個(gè)C++程序員,學(xué)習(xí)C#應(yīng)該不是一件很困難的事情,因?yàn)?/span>C#的許多特性都是從C++“繼承”(不精確的說(shuō)法)來(lái)的。但是C#作為一門全新的編程語(yǔ)言,它必然有自己的新特性,而這些C++中并不存在的新特性正是我們從C++轉(zhuǎn)向C#的過(guò)程中必須認(rèn)真重新學(xué)習(xí)的東西。“事件(event)”是C#的一個(gè)比較簡(jiǎn)單的新特性,我們今天就從“事件”開始,看一看C#的事件到底是怎么回事。

C#的事件和Windows窗口編程中提到的“事件”、“消息”、“事件驅(qū)動(dòng)”等在概念上是很類似的。我們?cè)谝粋€(gè)窗口上移動(dòng)鼠標(biāo)指針,系統(tǒng)就會(huì)產(chǎn)生WM_MOUSEMOVE消息(在vb中就是激發(fā)mousemove事件),只要我們告訴系統(tǒng)一個(gè)函數(shù)指針,系統(tǒng)就會(huì)通過(guò)這個(gè)回調(diào)函數(shù)通知我們,這是Windows窗口編程中的“事件”。C#中的事件的外延更廣,任何一個(gè)對(duì)象都可以擁有事件,客戶可以“定制”該對(duì)象的事件,當(dāng)該對(duì)象的內(nèi)部狀態(tài)發(fā)生特定的改變時(shí),就會(huì)通過(guò)定制事件時(shí)指定的函數(shù)代理(delegate)調(diào)用這個(gè)函數(shù)通知客戶。當(dāng)客戶不在需要事件通知時(shí),可以“撤消”對(duì)該事件的定制。

Now is the show time!模仿秀現(xiàn)在開始!

一、C#

我們先來(lái)看一個(gè)C#中一個(gè)“事件”的最簡(jiǎn)單例子:

VS.NET中新建一個(gè)C#ConsoleApplication,項(xiàng)目名稱為“eventtest”。為該項(xiàng)目新加一個(gè)類MyClass,對(duì)應(yīng)的源文件為MyClass.cs代碼如下:

using System;

namespace eventtest

{

    //定義EventHandler函數(shù)代理

    public delegate void EventHandler();

    /// <summary>

    ///

    /// </summary>

    public class MyClass

    {

        //構(gòu)造函數(shù)

        public MyClass(){}

        //聲明一個(gè)事件

        public eventEventHandler AEvent;

        //激發(fā)事件

        public voidFireEvent()

        {

            if(AEvent != null)

            {

                //直接把event當(dāng)做函數(shù)調(diào)用

                AEvent();

            }

        }

    }

}

 

public event EventHandlerAEvent;就是給MyClass定義了一個(gè)事件(通過(guò)event關(guān)鍵字),其事件處理函數(shù)(通知函數(shù))的原型由EventHandler函數(shù)代理(類似C++中的函數(shù)指針)指定。

FireEvent()成員函數(shù)用于激發(fā)該事件,如果客戶定制了該event,當(dāng)本類對(duì)象的FireEvent()公開方法被調(diào)用時(shí),客戶應(yīng)該可以得到通知。在Main函數(shù)里寫如下代碼:

using System;

namespace eventtest

{

    /// <summary>

    /// Class1 的摘要說(shuō)明。

    /// </summary>

    class Class1

    {

        /// <summary>

        /// 應(yīng)用程序的主入口點(diǎn)。

        /// </summary>

        [STAThread]

        static void Main(string[] args)

        {

            MyClassObj = new MyClass();

            Obj.AEvent+= new EventHandler(MyEventHandler);//定制事件

            Obj.FireEvent();//這行將導(dǎo)致MyEventHandler被調(diào)用

            Obj.AEvent-= new EventHandler(MyEventHandler);//撤消事件

            Obj.FireEvent();//這里將不會(huì)引發(fā)事件

           

            Console.WriteLine("結(jié)束!");

            Console.ReadLine();

        }

       

        //事件處理函數(shù)

        public static voidMyEventHandler()

        {

            Console.WriteLine("Thisis a event!");

        }

    }

   

}

首先寫一個(gè)具有適當(dāng)形式的事件處理(通知)函數(shù)MyEventHandler,然后通過(guò)Obj.AEvent+= new EventHandler(MyEventHandler)來(lái)定制事件。通過(guò)“-=”來(lái)撤消事件定制。

運(yùn)行程序我們可以發(fā)現(xiàn),當(dāng)客戶(Class1)定制了ObjAEvent事件后,在ObjFireEvent()成員函數(shù)被調(diào)用時(shí),客戶可以在MyEventHandler函數(shù)中得到通知(在這里只是簡(jiǎn)單地輸出一個(gè)文本)。而當(dāng)客戶撤消該事件的定制后,就不會(huì)再得到該事件通知。

 

二、C++

下面我們?cè)?span>C++中模擬該機(jī)制:

由于C++不支持event關(guān)鍵字,我們就必須自己寫代碼。在這里我通過(guò)模板類的手段來(lái)實(shí)現(xiàn),因?yàn)樵撌侄螌?shí)現(xiàn)的效果和C#比較類似。

VC6中新建一個(gè)win32console app,命名為“cppevent“。新建一個(gè).h頭文件,命名為“event.h”,代碼如下:

//event.h

template <typename Handler>

class event

{

private:

         Handler m_Handler;

protected:

         //模擬C# event add/remove訪問(wèn)器

         //如果要重新實(shí)現(xiàn)add/remove請(qǐng)?jiān)谂缮愔兄貙戇@兩個(gè)函數(shù)

         virtual void add(const Handler value){m_Handler = value;};

         virtual void remove(const Handler value){if(value == m_Handler)m_Handler = NULL;};

public:

         //構(gòu)造函數(shù)

         event():m_Handler(NULL){}

         //+= 操作符

         event& operator += (const Handler value)

         {

                   add(value);

                   return *this;

         }

         //-=操作符

         event& operator -= (const Handler value)

         {

                   remove(value);

                   return *this;

         }

         //PFN_EVENT_HANDLE 操作符

         operator Handler()

         {

                   return m_Handler;

         }

};

 

為了能夠在定義是指定事件處理函數(shù)的原型,我使用了template,為了能和C#一樣用+=-=來(lái)定制和撤消事件,我重載了這兩個(gè)操作符(C#不支持操作符重載),為了能像C#一樣直接把event當(dāng)做函數(shù)調(diào)用,我有重載了Handler自定義轉(zhuǎn)換操作符,可惜的是,這一點(diǎn)模擬得不是很像,在調(diào)用時(shí)還必須來(lái)一次強(qiáng)制轉(zhuǎn)換才可以:(,具體參看后面的代碼:

C++版的MyClass如下:

//MyClass.h

#include "event.h"

//定義EventHandler的函數(shù)指針類型

typedef void(*EventHandler)();

class MyClass

{

public:

         //構(gòu)造函數(shù)

         MyClass(){};

         //聲明一個(gè)事件

         event<EventHandler> AEvent;

         //激發(fā)事件

         void FireEvent()

         {

                   if(AEvent != NULL)

                   {

                            //C++中必須用EventHandler進(jìn)行強(qiáng)制類型轉(zhuǎn)換

                            ((EventHandler)AEvent)();

                   };

         }

};

 

C#版的MyClass比較一下你就會(huì)發(fā)現(xiàn)代碼非常接近,當(dāng)然,C#是在語(yǔ)言級(jí)直接支持event關(guān)鍵字的,而C++不支持,用模板類代替,所以聲明事件的代碼有些不一樣。還有就是FireEvent()C++不能把event對(duì)象直接當(dāng)做函數(shù)來(lái)調(diào)用,多了強(qiáng)制類型轉(zhuǎn)換。

 

C++版的客戶代碼如下:

//cppevent.cpp : Defines the entry point for the console application.

//

#include "stdafx.h"

#include "MyClass.h"

//向前聲明

void MyEventHandler();

int main(int argc, char* argv[])

{

         MyClass Obj;

         Obj.AEvent += MyEventHandler;//定制事件

         Obj.FireEvent();//這行將導(dǎo)致MyEventHandler被調(diào)用

         Obj.AEvent -= MyEventHandler;//撤消事件

         Obj.FireEvent();//這個(gè)將不會(huì)引發(fā)事件

         printf("結(jié)束!\n");

         char n;

         scanf("%c",&n);

         return 0;

}

void MyEventHandler()

{

         printf("Thisis a event!\n");

}

 

我們可以看到,可C#版的客戶代碼相比,核心部分是非常接近的,我們已經(jīng)可以和C#一樣用“+=”和“-=”來(lái)定制事件和撤消事件定制,并在ObjFireEvent()被調(diào)用時(shí)收到事件通知,輸出文本。

鑒于篇幅的原因,我們沒(méi)有仔細(xì)比較兩個(gè)版本的eventaddremove訪問(wèn)器/成員函數(shù),其實(shí)二者也是非常類似的,你可以自己試試。C++版的eventaddremove均為virtual的,你可以從event類繼承出來(lái)一個(gè)MyEvent類,然后重新實(shí)現(xiàn)這兩個(gè)函數(shù),就可以定制自己的addremove了。這和C#add/remove訪問(wèn)器的也是非常相像的。

 

三、總結(jié)

通過(guò)這場(chǎng)“模仿show”我們可以從更深的層次理解C#event機(jī)制,更重要的是我們用自己所熟悉的東西(C++,模板類)來(lái)模仿并解釋了我們目前還不太熟悉的東西(C#event)。

其實(shí),C#delegate就是C++的函數(shù)指針,C# event的核心機(jī)制就是C++中的模板(定義event時(shí)表現(xiàn)出來(lái))和運(yùn)算符重載(+=-=和直接把event當(dāng)做函數(shù)調(diào)用)的結(jié)合體。C#C++中容易出錯(cuò)的部分用“新特性”封裝了起來(lái),把這部分工作從programmer身上轉(zhuǎn)移到了compiler身上,讓我們把更多的精力集中到業(yè)務(wù)邏輯的處理上。

posted on 2008-06-30 06:02 RedLight 閱讀(686) 評(píng)論(0)  編輯 收藏 引用 所屬分類: C++功夫篇

<2008年4月>
303112345
6789101112
13141516171819
20212223242526
27282930123
45678910

導(dǎo)航

統(tǒng)計(jì)

公告


Name: Galen
QQ: 88104725

常用鏈接

留言簿(3)

隨筆分類

隨筆檔案

相冊(cè)

My Friend

搜索

最新評(píng)論

閱讀排行榜

評(píng)論排行榜

青青草原综合久久大伊人导航_色综合久久天天综合_日日噜噜夜夜狠狠久久丁香五月_热久久这里只有精品
  • <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>
            亚洲片在线资源| 西瓜成人精品人成网站| 亚洲一区免费视频| 日韩一级片网址| 一本色道久久加勒比88综合| 亚洲精品影院| 亚洲一区二区在线视频| 欧美一区二区三区在线| 久久精品国产在热久久 | av72成人在线| 亚洲午夜免费福利视频| 欧美一区二区日韩一区二区| 久久深夜福利免费观看| 欧美喷水视频| 国产欧美日韩一区二区三区在线| 韩国一区二区三区在线观看| 亚洲精品日本| 欧美一区二区性| 欧美大片一区| 一区二区三区视频免费在线观看| 欧美专区一区二区三区| 你懂的亚洲视频| 国产精品一二三| 亚洲欧洲精品一区二区精品久久久 | 欧美一区二区三区视频免费| 久久伊人一区二区| 一区二区免费在线播放| 久久在线免费观看| 国产精品亚洲一区二区三区在线| 91久久夜色精品国产九色| 午夜日韩激情| 亚洲精品一二| 久久亚洲精品视频| 国产一二三精品| 一区二区三区高清在线| 免费久久精品视频| 欧美亚洲视频在线观看| 欧美三级电影一区| 亚洲精品一区二区三| 久久综合九色综合欧美就去吻| 中国成人在线视频| 欧美另类极品videosbest最新版本| 国自产拍偷拍福利精品免费一| 亚洲欧美色婷婷| 91久久精品一区| 久久精品国产69国产精品亚洲| 国产精品久久久久久久久借妻| 999在线观看精品免费不卡网站| 免费久久99精品国产| 欧美一级大片在线观看| 国产精品亚洲精品| 午夜亚洲一区| 亚洲一区二区三区777| 欧美大片第1页| 久久精品国产99国产精品| 国产午夜一区二区三区| 欧美一区二区免费观在线| 亚洲在线播放| 国产精品欧美日韩久久| 亚洲欧美日韩国产精品| 一区二区三区产品免费精品久久75 | 一区二区三区高清不卡| 亚洲人成久久| 欧美日本韩国一区二区三区| 亚洲精品日韩欧美| 亚洲精品久久在线| 欧美巨乳在线| 国产精品99久久久久久久久| 99v久久综合狠狠综合久久| 欧美日韩精品国产| 亚洲一区二区三区涩| 亚洲特级片在线| 国产精品日韩精品欧美在线| 欧美伊人久久大香线蕉综合69| 欧美一级片久久久久久久| 国模一区二区三区| 欧美成人精品不卡视频在线观看| 久久精品视频免费观看| 欧美一站二站| 亚洲在线视频免费观看| 欧美日韩综合不卡| 性色av一区二区三区| 亚洲一区国产精品| 国产欧美精品日韩| 久久国产日韩| 美女主播一区| 亚洲自拍偷拍视频| 午夜国产精品视频免费体验区| 国产一区二区三区无遮挡| 欧美高清视频| 国产精品第三页| 久久成人一区二区| 久久精品成人一区二区三区蜜臀| 亚洲电影一级黄| 日韩视频一区二区三区在线播放| 国产精品久久久久久久久久三级| 欧美专区日韩专区| 欧美黄免费看| 久久精品免视看| 欧美大片一区二区| 久久国产精品久久国产精品| 欧美国产日产韩国视频| 香蕉成人伊视频在线观看| 久久香蕉国产线看观看av| 久久精品视频网| 中文久久精品| 久久亚洲国产精品一区二区| 亚洲综合视频1区| 老司机aⅴ在线精品导航| 亚洲一级免费视频| 欧美96在线丨欧| 久久久99国产精品免费| 欧美网站在线| 亚洲高清影视| 激情欧美一区二区三区| 一区二区欧美在线| 日韩一级黄色av| 久久久综合精品| 欧美一级精品大片| 国产精品国产a| 日韩视频中文字幕| 亚洲国产高清在线| 欧美一区二区视频97| 亚洲欧美在线免费| 欧美视频在线不卡| 亚洲精品人人| 亚洲美女精品久久| 免播放器亚洲| 欧美1区2区视频| 在线观看国产日韩| 可以看av的网站久久看| 老司机精品久久| 激情成人av| 久久色在线播放| 欧美www视频| 亚洲电影下载| 免费看的黄色欧美网站| 免费在线亚洲欧美| 激情欧美亚洲| 久久福利精品| 久久亚洲国产精品一区二区 | 欧美一区二区三区电影在线观看| 先锋a资源在线看亚洲| 国产麻豆视频精品| 亚洲欧美中文日韩v在线观看| 先锋影院在线亚洲| 国产亚洲欧美aaaa| 久久亚洲综合色一区二区三区| 欧美激情第一页xxx| 日韩香蕉视频| 国产精品久久一区二区三区| 欧美一级大片在线观看| 女女同性精品视频| 亚洲精品中文在线| 国产精品久久久久久亚洲调教| 亚洲免费视频网站| 久久久久国色av免费看影院| 亚洲黄色av| 欧美性大战久久久久久久| 亚洲欧美激情一区二区| 久久夜色精品| 99热精品在线观看| 国产欧美一区二区三区久久人妖| 久久国产一区二区| 亚洲国产婷婷综合在线精品| 亚洲午夜激情| 激情婷婷欧美| 欧美日韩精品中文字幕| 午夜精品福利视频| 欧美国产亚洲视频| 亚洲欧美电影在线观看| 一区二区自拍| 欧美日韩另类一区| 亚洲一区精品视频| 一区二区三区在线免费观看| 欧美精品成人在线| 欧美一区二区三区四区在线观看地址| 麻豆精品一区二区综合av | 久久国产精彩视频| 亚洲精品影视在线观看| 国产欧美韩日| 欧美日本乱大交xxxxx| 久久xxxx| av成人免费在线| 美女黄色成人网| 午夜精品一区二区在线观看 | 国产精品综合| 欧美激情在线| 久久精品国产亚洲一区二区| 国产精品99久久久久久久久| 欧美国产视频在线| 久热精品视频在线免费观看| 亚洲一区尤物| 99精品欧美| 亚洲精美视频| 激情综合网址| 国产视频一区在线观看| 国产精品美女视频网站| 欧美午夜无遮挡| 欧美日韩国产综合在线|