delegate,中文不知道怎么翻譯,姑且叫代理吧。
delegate 的概念其實(shí)早就出現(xiàn),只不過(guò)我看到它的時(shí)候卻是在研究Windows 8時(shí),查閱網(wǎng)上的資料,delegate應(yīng)該是最先出現(xiàn)在.NET框架中,為的是解決函數(shù)指針的不安全性。
為什么說(shuō)c++的函數(shù)指針不安全呢?想到函數(shù)指針你能想到什么?不錯(cuò),函數(shù)的地址,函數(shù)的入口點(diǎn),然后呢?沒(méi)了,確實(shí)就是這些東西,函數(shù)指針即沒(méi)有表示這個(gè)函數(shù)有什么返回類型,也沒(méi)有指示這個(gè)函數(shù)有什么形式的參數(shù),更沒(méi)有指示有幾個(gè)參數(shù),你可以隨意轉(zhuǎn)換這個(gè)函數(shù)指針,函數(shù)指針是非安全的。
那么,既然要升級(jí)C++,那就為函數(shù)指針的安全性做點(diǎn)修改吧。你不是只有個(gè)地址嗎,沒(méi)關(guān)系,我給你封裝一下,封裝成一個(gè)類,這個(gè)類里面你想要的關(guān)于這個(gè)函數(shù)的所有信息都有,甚至這個(gè)函數(shù)被哪個(gè)對(duì)象調(diào)用我都封裝進(jìn)去。好了吧。
同其他的類一樣,delegate就是一個(gè)數(shù)據(jù)結(jié)構(gòu),你可以在把它放在類中作為類的數(shù)據(jù)成員,也可以當(dāng)成形參,其他類可以怎么折騰,你就可以怎么折騰,從此你就不是一個(gè)地址了,你也是可以和別人平起平坐的類了。哇哈哈,高興壞了。
言歸正傳,這里參考MSDN中的官方文檔來(lái)說(shuō)明一下delegate:
原文鏈接地址:
http://msdn.microsoft.com/en-us/library/windows/apps/hh441500(v=vs.110).aspx 下面是一個(gè)delegate的聲明:如果你是在研究Windows 8 的話,^(姑且叫它hat吧)標(biāo)識(shí)符你肯定比較關(guān)注,這是說(shuō)明這個(gè)對(duì)象是Windows Runtime 類型的,這個(gè)對(duì)象不用你自己去release,他會(huì)采用引用計(jì)數(shù)技術(shù),當(dāng)你不需要的時(shí)候,后臺(tái)自動(dòng)給你釋放掉,類似于Java的垃圾回收機(jī)制,其實(shí)就是把原來(lái)的COM中的Release方法給封裝了一下。言歸正傳,這個(gè)delegate聲明表示,它封裝了一個(gè)方法,這個(gè)方法的返回值是Platform::String^ ,參數(shù)也是Platform::String^(這個(gè)我就不解釋了,都是WinRT里面的新東西,大家WinRT稍微用用就會(huì)經(jīng)常接觸這個(gè)東西,不同于std::string)
下面又定義了一個(gè)借口,我們可以看到NormalizeZipCode^ normFunc可以作為形參。
下面我們定義一個(gè)類來(lái)實(shí)現(xiàn)這個(gè)接口:
1 ref class ZipCodes: public IZipCodes
2 {
3 private:
4 Platform::String^ LookupFromCityState(Platform::String^ city, Platform::String^ state);
5 Platform::String^ Normalize(Platform::String^ zip);
6 public:
7 Platform::String^ ZipCodeFromCity(Platform::String^ city, Platform::String^ state,
8 NormalizeZipCode^ normFunc)
9 {
10 auto zip = LookupFromCityState(city, state);
11 NormalizeZipCode^ func = (normFunc != nullptr ? normFunc :
12 ref new NormalizeZipCode(this, &ZipCodes::Normalize));//這里是重點(diǎn),給func賦值,可以賦給它一個(gè)nullptr指針,也可以new一個(gè)給它。
13 return func(zip);
14 }
15 };
我們看到這個(gè)ZipCode類實(shí)現(xiàn)了該接口,并實(shí)現(xiàn)了這個(gè)方法。我們可以這樣賦值:NormalizeZipCode ^ func = ref new NormalizeZipCode(&ZipCodes::Normalize);
調(diào)用嘛,大家也看到了,就直接用func(zip)就可以了,這時(shí)候,如果傳輸?shù)膎ormFunc是空值,那么就給func賦值為
NormalizeZipCode(this, &ZipCodes::Normalize) ,也就是說(shuō)這時(shí)的func就是Normalize函數(shù)的代理了,可以直接用func(zip)來(lái)調(diào)用Normalize(。。。),或許這就是為什么這個(gè)類型為什么叫代理類型的原因吧。
使用上述的類和方法的時(shí)候,就跟其他類一樣,直接用就行了。
1 IZipCodes ^codes = ref new ZipCodes();2 auto func = ref new NormalizeZipCode([](Platform::String^ zipcode) {//lambda 表達(dá)式
3 return "98103";
4 });
5
6 auto zip1 = codes->ZipCodeFromCity("Seattle", "WA", nullptr);//這里調(diào)用哪個(gè)?
7 auto zip2 = codes->ZipCodeFromCity("Seattle", "WA", func);// 這里調(diào)用哪個(gè)?
8
上述的[]應(yīng)該是lambda表達(dá)式,我沒(méi)有深究。有了解的朋友請(qǐng)給我補(bǔ)補(bǔ)。
posted on 2012-02-14 15:19
Dino-Tech 閱讀(293)
評(píng)論(0) 編輯 收藏 引用