• <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>

            為生存而奔跑

               :: 首頁(yè) :: 聯(lián)系 :: 聚合  :: 管理
              271 Posts :: 0 Stories :: 58 Comments :: 0 Trackbacks

            留言簿(5)

            我參與的團(tuán)隊(duì)

            搜索

            •  

            積分與排名

            • 積分 - 330187
            • 排名 - 74

            最新評(píng)論

            閱讀排行榜

            評(píng)論排行榜

            眾所周知,STL使用起來(lái)非常方便,其中仿函數(shù)(functor)扮演了一個(gè)非常重要的角色。靈活運(yùn)用仿函數(shù)的使用對(duì)于發(fā)揮STL強(qiáng)大功能非常關(guān)鍵。本文詳細(xì)介紹了如何使用mem_fun和mem_fun1來(lái)綁定類(lèi)成員函數(shù),使之成為functor

            什么是仿函數(shù)?就是一個(gè)重載了"()"運(yùn)算符的struct,例如:

            struct print_obj{
               
            void operator(int a)const{
                    
            cout<<a<<endl;
                }
            };

            在STL的許多算法(algorithm)中都需要使用functor. 如:for_each. 同樣在關(guān)聯(lián)容器中也需要使用functor, 如map, set等。經(jīng)常在使用STL算法的時(shí)候,經(jīng)常需要把仿函數(shù)和類(lèi)聯(lián)系在一起,如果可以直接使用類(lèi)的成員函數(shù)作為仿函數(shù),那就方便多了。mem_fun的功能就是如此。

            先看個(gè)簡(jiǎn)單的例子:

            struct D {
              D(int i=0){num=i;}
              int num;
            };
            struct print_D{
             
            void operator()(const D* d)const{
                 
            cout<<"I am D. my num="<<d->num<<endl;
                }
            };

            int main()
            {
              vector<D*> V;

              V.push_back(new D(1));
              V.push_back(new D(2));
              V.push_back(new D);
              V.push_back(new D(3));

              for_each(V.begin(), V.end(), print_D());
            }
            編譯輸出:

            I am D. my num=1
            I am D. my num=2
            I am D. my num=0
            I am D. my num=3

            如果使用mem_fun,會(huì)方便很多:

            struct D {
              D(int i=0){num=i;}
              void print() { cout << "I'm a D. my num=" << num<< endl; }
              int num;
            };

            int main()
            {
              vector<D*> V;

              V.push_back(new D(1));
              V.push_back(new D(2));
              V.push_back(new D);
              V.push_back(new D(3));

              for_each(V.begin(), V.end(), mem_fun(&D::print));
            }

            是不是省了一個(gè)仿函數(shù)?方便多了,沒(méi)錯(cuò)吧。這也更符合面向?qū)ο蟮囊?guī)則。不過(guò)這樣好像讓人難以理解,這里告訴你一個(gè)理解STL的訣竅:

                如果對(duì)STL的某個(gè)部分不了解,就去看源碼,源碼是最好的老師。

            那看看源碼是怎么回事,在SGI STL的stl_function.h:

            template <class _Ret, class _Tp>
            inline mem_fun_t<_Ret,_Tp> mem_fun(_Ret (_Tp::*__f)())
            { return mem_fun_t<_Ret,_Tp>(__f); }

            原來(lái)mem_fun返回的是一個(gè)對(duì)象:mem_fun_t<_Ret,_Tp>.(不要嫌人家命名太怪異).那mem_fun_t<_Ret,_Tp>又是什么東東?還是看源碼:

            template <class _Ret, class _Tp>
            class mem_fun_t : public unary_function<_Tp*,_Ret> {
            public:
               
            explicit mem_fun_t(_Ret (_Tp::*__pf)()) : _M_f(__pf) {}
                _Ret operator()(_Tp* __p)
            const { return (__p->*_M_f)(); }
            private:
                _Ret (_Tp::*_M_f)();
            };

            看明白了嗎?原來(lái)mem_fun_t就是一個(gè)functor,這下就滿(mǎn)足了for_each的要求了。其調(diào) 用流程是這樣的,for_each把vector中的元素傳送給mem_fun,mem_fun自己產(chǎn)生一個(gè)仿函數(shù)mem_fun_t,然后仿函數(shù)調(diào)用其 重載的()。過(guò)程就這么簡(jiǎn)單。當(dāng)然你不能對(duì)其他類(lèi)的成員函數(shù)進(jìn)行綁定,因?yàn)樵趂or_each調(diào)用過(guò)程中,會(huì)傳遞其*iterator值,如果是其他類(lèi)的 成員函數(shù),那么這個(gè)類(lèi)的對(duì)象無(wú)法傳入,當(dāng)然就無(wú)法完成任務(wù)了。

            這里使用的是vector<D*> V; 在mem_fun_t構(gòu)造函數(shù)中,剛好需要指針,如果不是D*, 而是使用vector<D> V; 還能用嗎?

            這是你需要使用的是mem_fun_ref。把程序改成:

            struct D {
              D(int i=0){num=i;}
              void print() { cout << "I'm a D. my num=" << num<< endl; }
              int num;
            };

            int main()
            {
              vector<D> V;

              V.push_back(D(1));
              V.push_back( D(2));
              V.push_back( D());
              V.push_back( D(3));

              for_each(V.begin(), V.end(), mem_fun_ref(&D::print));
            }

            一切都OK了。

            mem_fun對(duì)于一些多態(tài)的虛函數(shù)也十分有用,注意看下面的例子:

            struct B {
              virtual void print() = 0;
            };

            struct D1 : public B {
              void print() { cout << "I'm a D1" << endl; }
            };

            struct D2 : public B {
              void print() { cout << "I'm a D2" << endl; }
            };

            int main()
            {
              vector<B*> V;

              V.push_back(new D1);
              V.push_back(new D2);
              V.push_back(new D2);
              V.push_back(new D1);

              for_each(V.begin(), V.end(), mem_fun(&B::print));
            }

            posted on 2009-12-01 18:31 baby-fly 閱讀(681) 評(píng)論(0)  編輯 收藏 引用 所屬分類(lèi): Effective STL / C++
            久久久久这里只有精品| 国产精品九九久久精品女同亚洲欧美日韩综合区| 精品久久久无码人妻中文字幕| 狠狠色婷婷久久一区二区 | 久久久久亚洲精品日久生情| 久久人人超碰精品CAOPOREN| 一本色道久久综合狠狠躁| 国产精品日韩欧美久久综合| 国内精品久久久久久久久电影网 | 精品国产乱码久久久久久呢| 国产高清美女一级a毛片久久w| A级毛片无码久久精品免费| 久久国产视频网| 久久香蕉一级毛片| 久久精品国产半推半就| 99久久国产热无码精品免费| 一个色综合久久| 亚洲精品97久久中文字幕无码| 国产精品无码久久久久| 日本福利片国产午夜久久| 国内精品久久九九国产精品| 久久精品国产亚洲AV电影| 99久久99这里只有免费费精品 | 久久久久久无码Av成人影院| 日韩人妻无码精品久久久不卡| 亚洲精品美女久久久久99| 久久久久亚洲av无码专区导航 | 久久99亚洲综合精品首页| 亚洲人成无码久久电影网站| 国产精品久久婷婷六月丁香| 久久99热只有频精品8| 欧美日韩中文字幕久久久不卡| 色青青草原桃花久久综合| 久久久久成人精品无码中文字幕| 久久er国产精品免费观看8| 久久热这里只有精品在线观看| 国产精品久久毛片完整版| 99精品国产免费久久久久久下载| 国产精品一区二区久久| 久久久久久久97| 久久精品人人做人人爽电影|