• <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>
            Impossible is nothing  
              愛過知情重醉過知酒濃   花開花謝終是空   緣份不停留像春風(fēng)來又走   女人如花花似夢(mèng)
            公告
            日歷
            <2025年6月>
            25262728293031
            1234567
            891011121314
            15161718192021
            22232425262728
            293012345
            統(tǒng)計(jì)
            • 隨筆 - 8
            • 文章 - 91
            • 評(píng)論 - 16
            • 引用 - 0

            導(dǎo)航

            常用鏈接

            留言簿(4)

            隨筆分類(4)

            隨筆檔案(8)

            文章分類(77)

            文章檔案(91)

            相冊(cè)

            搜索

            •  

            最新評(píng)論

            閱讀排行榜

            評(píng)論排行榜

             
            C語(yǔ)言的指針相當(dāng)?shù)撵`活方便,但也相當(dāng)容易出錯(cuò)。許多C語(yǔ)言初學(xué)者,甚至C語(yǔ)言老鳥都很容易栽倒在C語(yǔ)言的指針下。但不可否認(rèn)的是,指針在C語(yǔ)言中的位置極其重要,也許可以偏激一點(diǎn)的來說:沒有指針的C程序不是真正的C程序。
               然而C++的指針卻常常給我一種束手束腳的感覺。C++比C語(yǔ)言更加強(qiáng)調(diào)強(qiáng)類型,強(qiáng)調(diào)類型安全,強(qiáng)調(diào)編譯時(shí)檢查。因此,對(duì)于C語(yǔ)言中最容易錯(cuò)用的指針, 更是不能放過:C++的指針被分成數(shù)據(jù)指針,數(shù)據(jù)成員指針,函數(shù)指針,成員函數(shù)指針,而且不能隨便相互轉(zhuǎn)換。而且這些指針的聲明格式都不一樣:

            數(shù)據(jù)指針 T *
            成員數(shù)據(jù)指針 T::*
            函數(shù)指針 R (*)(...)
            成員函數(shù)指針 R (T::*)(...)

              盡管C++中仍然有萬能指針void*,但它卻屬于被批斗的對(duì)象,而且再也不能“萬能”了。它不能轉(zhuǎn)換成成員指針。

              這樣一來,C++的指針就變得很尷尬:我們需要一種指針能夠指向同一類型的數(shù)據(jù),不管這個(gè)數(shù)據(jù)是普通數(shù)據(jù),還是成員數(shù)據(jù);我們更需要一種指針能夠指向同一類型的函數(shù),不管這個(gè)函數(shù)是靜態(tài)函數(shù),還是成員函數(shù)。但是沒有,至少?gòu)默F(xiàn)在的C++標(biāo)準(zhǔn)中,還沒有看到。
             
            沐楓網(wǎng)志 C++指針探討(三)成員函數(shù)指針

              自從有了類,我們開始按照 數(shù)據(jù)+操作 的方式來組織數(shù)據(jù)結(jié)構(gòu);自從有了模板,我們又開始把 數(shù)據(jù) 和 算法 分離,以便重用,實(shí)在夠折騰人的。但不管怎么折騰,現(xiàn)在大多數(shù)函數(shù)都不再單身,都嫁給了類,進(jìn)了圍城??墒俏覀?nèi)匀恍枰軌蜃杂烧{(diào)用這些成員函數(shù)。
              考慮一下windows下的定時(shí)調(diào)用。SetTimer函數(shù)的原型是這樣的:

            UINT_PTR SetTimer(
                HWND hWnd,
                UINT_PTR nIDEvent,
                UINT uElapse,
                TIMERPROC lpTimerFunc
            );
               其中,參數(shù)就不解釋了,這個(gè)函數(shù)估計(jì)大多數(shù)windows開發(fā)人員都知道。lpTimerFunc是個(gè)會(huì)被定時(shí)調(diào)用的函數(shù)指針。假如我們不通過 WM_TIMER消息來觸發(fā)定時(shí)器,而是通過lpTimerFunc來定時(shí)工作,那么我們就只能使用普通函數(shù)或靜態(tài)函數(shù),而無論如何都不能使用成員函數(shù), 哪怕通過靜態(tài)函數(shù)轉(zhuǎn)調(diào)也不行。

              再考慮一下線程的創(chuàng)建:
            uintptr_t _beginthread( 
               
            void*start_address )( void * ),
               unsigned stack_size,
               
            void *arglist 
            );
               start_address仍然只支持普通函數(shù)。不過這回好了,它允許回調(diào)函數(shù)一個(gè)void*參數(shù),它將會(huì)arglist作為參數(shù)來調(diào)用 start_address。于是,聰明的C++程序員,就利用arglist傳遞this指針,從而利用靜態(tài)函數(shù)成功的調(diào)用到了成員函數(shù)了:
            class mythread
            {
              
            public:
                
            static void doit(void* pThis)
                
            {
                ((mythread*)pThis)
            ->doit();
                }

                
            void doit(){}
            }
            ;

            main()
            {
              
              mythread
            * pmt = new mythread;
              _beginthread(
            &mythread::doit, 0, (void*)pmt);
              
            }

              但是顯然,C++程序員肯定不會(huì)因此而滿足。這里頭有許多被C++批判的不安定因素。它使用了C++中被認(rèn)為不安全的類型轉(zhuǎn)換,不安全的 void*指針,等等等等。但這是系統(tǒng)為C語(yǔ)言留下的調(diào)用接口,這也就認(rèn)了。那么假如,我們就在C++程序中如何來調(diào)用成員函數(shù)指針呢?
              如下例,我們打算對(duì)vector中的所有類調(diào)用其指定的成員函數(shù):

            #include <vector>
            #include 
            <algorithm>
            #include 
            <functional>
            #include 
            <iostream>
            using namespace std;

            class A
            {
                
            int value;
            public:
                A(
            int v){value = v;}
                
            void doit(){ cout << value << endl;};
                
            static void call_doit(A& rThis)
                
            {
                    rThis.doit();
                }

            }
            ;


            int main()
            {
                vector
            <A> va;
                va.push_back(A(
            1));
                va.push_back(A(
            2));
                va.push_back(A(
            3));
                va.push_back(A(
            4));
                
            //方法1:
                
            //for_each(va.begin(), va.end(), &A::doit); //error
                
            //方法2:
                for_each(va.begin(), va.end(), &A::call_doit);
                
            //方法3:
                for_each(va.begin(), va.end(), mem_fun_ref<void, A>(&A::doit));

                system(
            "Pause");

                
            return 0;
            }

              方法1,編譯不能通過。for_each只允許具有一個(gè)參數(shù)的函數(shù)指針或函數(shù)對(duì)象,哪怕A::doit默認(rèn)有一個(gè)this指針參數(shù)也不行。不是for_each沒考慮到這一點(diǎn),而是根本做不到!
              方法2,顯然是受到了beginthread的啟發(fā),使用一個(gè)靜態(tài)函數(shù)來轉(zhuǎn)調(diào)用,哈哈成功了。但是不爽!這不是C++。
              方法3,呼,好不容易啊,終于用mem_fun_ref包裝成功了成員函數(shù)指針。
               似乎方法3不錯(cuò),又是類型安全的,又可以通用--慢著,首先,它很丑,哪有調(diào)用普通C函數(shù)指針那么漂亮?。ㄒ姺椒?),用了一大串包裝,又是尖括號(hào)又是 圓括號(hào),還少不了&號(hào)!其次,它只能包裝不超過一個(gè)參數(shù)的函數(shù)!盡管它在for_each中夠用了,但是你要是想用在超過一個(gè)參數(shù)的場(chǎng)合,那只有 一句話:不可能的任務(wù)。

              是的,在標(biāo)準(zhǔn)C++中,這是不可能的任務(wù)。但事情并不總是悲觀的,至少有許多第三方庫(kù)提供了超越mem_fun的包裝。如boost:: function等等。但是它也有限制:它所支持的參數(shù)仍然是有限的,只有十多個(gè),盡管夠你用的了;同樣,它也是丑陋的,永遠(yuǎn)不要想它能夠簡(jiǎn)單的用 &來搞定。

              也許,以失去美麗的代價(jià),來?yè)Q取質(zhì)量上的保證,這也是C++對(duì)于函數(shù)指針的一種無奈吧……

              期待C++0x版本。它通過可變模板參數(shù),能夠讓mem_fun的參數(shù)達(dá)到無限個(gè)……
            posted on 2006-03-13 22:37 笑笑生 閱讀(220) 評(píng)論(0)  編輯 收藏 引用 所屬分類: C++語(yǔ)言
             
            Copyright © 笑笑生 Powered by: 博客園 模板提供:滬江博客
            一本色道久久88综合日韩精品 | 蜜臀久久99精品久久久久久| 91久久婷婷国产综合精品青草| 久久精品aⅴ无码中文字字幕不卡| 久久综合久久自在自线精品自| 国产一久久香蕉国产线看观看| 久久久久久久久久免免费精品| 久久99久国产麻精品66| 久久ZYZ资源站无码中文动漫| 精品久久777| 久久男人AV资源网站| 久久久老熟女一区二区三区| 国产国产成人久久精品| 伊人久久综合无码成人网| 精品久久久久一区二区三区| 久久精品国产清高在天天线| 九九热久久免费视频| 久久精品国产亚洲av高清漫画 | 久久强奷乱码老熟女网站| 婷婷综合久久中文字幕蜜桃三电影| 久久久免费观成人影院| 成人妇女免费播放久久久| 亚洲日本va中文字幕久久| 国内精品久久久久久久coent| 久久综合狠狠综合久久| 综合久久久久久中文字幕亚洲国产国产综合一区首 | 亚洲狠狠婷婷综合久久久久| 国产精久久一区二区三区| 久久久噜噜噜久久熟女AA片| 精品久久久久久中文字幕大豆网| 国产成人精品综合久久久| 91久久精品91久久性色| 国内精品久久人妻互换| 精品久久久无码人妻中文字幕豆芽| 亚洲婷婷国产精品电影人久久| 久久黄视频| 一本久久免费视频| 久久亚洲AV无码精品色午夜| 亚洲狠狠婷婷综合久久蜜芽 | 久久超乳爆乳中文字幕| 久久婷婷成人综合色综合|