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

C++ Programmer's Cookbook

{C++ 基礎(chǔ)} {C++ 高級} {C#界面,C++核心算法} {設(shè)計模式} {C#基礎(chǔ)}

引領(lǐng)boost(六)(boost::bind)

 

Boost::bind


一 Boost::bind
   
   在STL中,我們經(jīng)常需要使用bind1st,bind2st函數(shù)綁定器和fun_ptr,mem_fun等函數(shù)適配器,這些函數(shù)綁定器和函數(shù)適配器使用起來比較麻

煩,需要根據(jù)是全局函數(shù)還是類的成員函數(shù),是一個參數(shù)還是多個參數(shù)等做出不同的選擇,而且有些情況使用STL提供的不能滿足要求,所以如

果可以我們最好使用boost提供的bind,它提供了統(tǒng)一的接口,提供了更多的支持,比如說它增加了shared_ptr,虛函數(shù),類成員的綁定。

二 源碼剖析

 1) bind1st,bind2st函數(shù)綁定器,把二元函數(shù)對象變?yōu)橐辉瘮?shù)對象。
 2) mem_fun,把成員函數(shù)變?yōu)楹瘮?shù)對象。
 3) fun_ptr,把一般的全局函數(shù)變?yōu)楹瘮?shù)對象。
 4) boost::bind(),包含了以上所有的功能。


三 實例

 1)區(qū)別與mem_fun和fun_ptr

#include <functional>
#include 
<iostream>
#include 
<string>
#include 
"boost/bind.hpp"
class some_class 
{
public:      
    
void print_string(const std::string& s) const
    
{    
        std::cout 
<< s << '\n'
    }

    
void print_classname()
    
{
        std::cout 
<< "some_class" << std::endl;
    }

}
;
void print_string(const std::string s) 
{  std::cout << s << '\n';
}

void print_functionname()
{
    std::cout 
<< "Print_functionname" <<std::endl;
}

int main() 
{  
    std::ptr_fun(
&print_string)("hello1");
    
//std::ptr_fun<void>(&print_functionname);
    some_class sc0;
    std::mem_fun_ref(
&some_class::print_classname)(sc0);
    std::mem_fun_ref
<void,some_class>(&some_class::print_classname)(sc0);
    
//std::mem_fun1_ref<void,some_class,const std::stirng>(&some_class::print_string)(sc0,"hello2");

    (boost::bind(
&print_string,_1))("Hello func!");  
    boost::bind(
&print_functionname);
    some_class sc;  
    (boost::bind(
&some_class::print_classname,_1)(sc));
    (boost::bind(
&some_class::print_string,_1,_2))(sc,"Hello member!");
}

 

 2)區(qū)別與bind1st和bind2st

#include <functional>
#include 
<iostream>
#include 
<string>
#include 
<vector>
#include 
<algorithm>
#include 
"boost/bind.hpp"
void main()
{
    std::vector
<int> ints;
    ints.push_back(
7);
    ints.push_back(
4);
    ints.push_back(
12);
    ints.push_back(
10);
    
int count=std::count_if(ints.begin(),  
        ints.end(), 
        boost::bind(std::logical_and
<bool>(),boost::bind(std::greater<int>(),_1,5),boost::bind(std::less_equal<int>(),_1,10))
        );
    std::cout 
<< count << '\n';
    std::vector
<int>::iterator int_it=std::find_if(ints.begin(),  
        ints.end(),  
        boost::bind(std::logical_and
<bool>(),boost::bind(std::greater<int>(),_1,5),boost::bind(std::less_equal<int>(),_1,10))
        );
    
if (int_it!=ints.end()) 
    
{  std::cout << *int_it << '\n';}

}

 

 3)區(qū)別傳ref和傳instance

// bind instance or reference
#include <functional>
#include 
<iostream>
#include 
<string>
#include 
<vector>
#include 
<algorithm>
#include 
"boost/bind.hpp"
class tracer 
{
public
    tracer() 
{    std::cout << "tracer::tracer()\n";  } 
    tracer(
const tracer& other) {    std::cout << "tracer::tracer(const tracer& other)\n";  } 
    tracer
& operator=(const tracer& other)
    
{    std::cout <<      "tracer& tracer::operator=(const tracer& other)\n";    return *this;  } 
    
~tracer() {    std::cout << "tracer::~tracer()\n"
    }
 
    
void print(const std::string& s) const
    
{    std::cout << s << '\n';  }
}
;

void main()
{
    tracer t;
    boost::bind(
&tracer::print,t,_1)(std::string("I'm called on a copy of t\n"));
    tracer t1;
    boost::bind(
&tracer::print,boost::ref(t1),_1)(  std::string("I'm called directly on t\n"));

}

 

 4)綁定虛函數(shù)

//bind vitual class function
#include <functional>
#include 
<iostream>
#include 
<string>
#include 
<vector>
#include 
<algorithm>
#include 
"boost/bind.hpp"
class base {
public:
    
virtual void print() const 
    
{    std::cout << "I am base.\n";  
    }
  
    
virtual ~base() {}
}
;
class derived : public base 
{
public
    
void print()const 
    
{    std::cout << "I am derived.\n";  }
}
;

void main()
{
    derived d;
    
base b;
    boost::bind(
&base::print,_1)(b);
    boost::bind(
&base::print,_1)(d);
}

 

 5)綁定成員變量

// bind class's member
#include <functional>
#include 
<iostream>
#include 
<string>
#include 
<vector>
#include 
<algorithm>
#include 
"boost/bind.hpp"
class personal_info 

    std::
string name_;  
    std::
string surname_;
    unsigned 
int age_;
public
    personal_info(
const std::string& n,const std::string& s,unsigned int age):name_(n),surname_(s),age_(age) {} 
    std::
string name() const {return name_;} 
    std::
string surname() const {return surname_;} 
    unsigned 
int age() const {return age_;}
}
;

void main()
{
 std::vector
<personal_info> vec;
 vec.push_back(personal_info(
"Little","John",30));
 vec.push_back(personal_info(
"Friar""Tuck",50));
 vec.push_back(personal_info(
"Robin""Hood",40));
 std::sort(vec.begin(),
     vec.end(),
     boost::bind(std::less
<unsigned int>(),boost::bind(&personal_info::age,_1),boost::bind(&personal_info::age,_2))
     );
 std::sort(vec.begin(),
     vec.end(),
     boost::bind(std::less
<std::string>(),boost::bind(&personal_info::surname,_1),boost::bind(&personal_info::surname,_2))
     );
}

 

四 注意

 1) 現(xiàn)在的類庫最多可以支持9個參數(shù)。
 2)在綁定一個成員函數(shù)時,bind 表達(dá)式的第一個參數(shù)必須是成員函數(shù)所在類的實例!理解這個規(guī)則的最容易的方法是,這個顯式的參數(shù)將取

替隱式的 this ,被傳遞給所有的非靜態(tài)成員函數(shù)。細(xì)心的讀者將會留意到,實際上這意味著對于成員函數(shù)的綁定器來說,只能支持八個參數(shù)

,因為第一個要用于傳遞實際的對象。
 3)當(dāng)我們傳遞某種類型的實例給一個 bind 表達(dá)式時,它將被復(fù)制,除非我們顯式地告訴 bind 不要復(fù)制它。要避免復(fù)制,我們必須告訴

bind 我們想傳遞引用而不是它所假定的傳值。我們要用 boost::ref 和 boost::cref (分別用于引用和 const 引用)來做到這一點,它們也是

Boost.Bind 庫的一部分。還有一種避免復(fù)制的方法;就是通過指針來傳遞參數(shù)而不是通過值來傳遞。
 4) 通過 Boost.Bind, 你可以象使用非虛擬函數(shù)一樣使用虛擬函數(shù),即把它綁定到最先聲明該成員函數(shù)為虛擬的基類的那個虛擬函數(shù)上。這

個綁定器就可以用于所有的派生類。如果你綁定到其它派生類,你就限制了可以使用這個綁定器的類。
 5)bind還可以綁定成員變量。

五 參考

1)Beyond the C++ Standard Library: An Introduction to Boost
2)boost在線document

 

posted on 2007-09-05 14:40 夢在天涯 閱讀(6744) 評論(4)  編輯 收藏 引用 所屬分類: CPlusPlusSTL/Boost

評論

# re: 引領(lǐng)boost(六)(boost::bind) 2007-09-05 14:41 夢在天涯

在第一個sample代碼中2行注釋的代碼有問題,誰能幫忙解決哦,請高手指點哦!  回復(fù)  更多評論   

# re: 引領(lǐng)boost(六)(boost::bind) 2007-09-05 16:52 螞蟻終結(jié)者

//std::ptr_fun<void>(&print_functionname);
std::ptr_fun返回的是unary function 或 binary function,不能用于無參函數(shù),實際上也沒必要。

//std::mem_fun1_ref<void,some_class,const std::stirng>(&some_class::print_string)(sc0,"hello2");
如果你用的是Macrosoft的STL,std::mem_fun1_ref會有bug,即只能用于非const成員函數(shù),如果把some_class中的
void print_string(const string& s) const
改為
void print_string(const string& s)

然后再這樣寫就可以了:
std::mem_fun1_ref(&some_class::print_string)(sc0, "hello2");

當(dāng)然了,用SGI的STL不會有這個bug
最好的辦法是用std::mem_fun_ref,std::mem_fun_ref可用于一個參數(shù)或零個參數(shù)的const或non-const成員函數(shù),std::mem_fun1_ref只能用于一個參數(shù)的const或non-const成員函數(shù),估計是為了兼容性。
所以也可以這樣寫:
std::mem_fun_ref(&some_class::print_string)(sc0, "hello2");
這樣print_string加不加const都一樣
  回復(fù)  更多評論   

# re: 引領(lǐng)boost(六)(boost::bind) 2007-09-05 17:01 夢在天涯

螞蟻終結(jié)者說的是正確的

std::mem_fun_ref(&some_class::print_string)(sc0,"hello2");
是正確的!(主要是我一開始以為它只能用于無參數(shù)函數(shù)的情況,看來是錯誤的,可以用于有一個參數(shù)的函數(shù))

//std::ptr_fun<void>(&print_functionname);
這個可能也想螞蟻終結(jié)者說的,沒有什么必要!

不知道stl里有沒有提供更多參數(shù)的支持!  回復(fù)  更多評論   

# re: 引領(lǐng)boost(六)(boost::bind) 2007-09-27 18:28 沐楓

@夢在天涯
有更多的參數(shù)支持。但默認(rèn)只支持10個以內(nèi)。
需要更多的參數(shù),要修改源代碼中的一個宏定義。我以前曾看過,代碼中有50個以內(nèi)的參數(shù)定義預(yù)留。
  回復(fù)  更多評論   

公告

EMail:itech001#126.com

導(dǎo)航

統(tǒng)計

  • 隨筆 - 461
  • 文章 - 4
  • 評論 - 746
  • 引用 - 0

常用鏈接

隨筆分類

隨筆檔案

收藏夾

Blogs

c#(csharp)

C++(cpp)

Enlish

Forums(bbs)

My self

Often go

Useful Webs

Xml/Uml/html

搜索

  •  

積分與排名

  • 積分 - 1812969
  • 排名 - 5

最新評論

閱讀排行榜

青青草原综合久久大伊人导航_色综合久久天天综合_日日噜噜夜夜狠狠久久丁香五月_热久久这里只有精品
  • <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>
              这里只有精品丝袜| 亚洲性线免费观看视频成熟| 久久精品五月| 伊人久久大香线| 狂野欧美一区| 免费欧美视频| 夜夜嗨av一区二区三区中文字幕| 亚洲精品乱码久久久久| 久久久久一区二区| 亚洲精品乱码久久久久久蜜桃91 | 亚洲韩国日本中文字幕| 欧美激情一区二区三区全黄 | 亚洲一区二区免费视频| 国产精品一区二区三区四区五区| 久久久精品动漫| 国产精品视频精品| 尹人成人综合网| 最新国产精品拍自在线播放| 欧美视频一二三区| 久久成人精品无人区| 久久久中精品2020中文| 99国产精品久久久久老师| 亚洲一级片在线观看| 韩国三级电影久久久久久| 亚洲第一福利在线观看| 国产精品久久久久久久7电影| 久久久久高清| 欧美日韩国产在线| 久久男女视频| 国产精品h在线观看| 久久婷婷影院| 欧美日韩高清在线观看| 巨乳诱惑日韩免费av| 欧美日韩国产精品专区| 久久夜色精品亚洲噜噜国产mv| 欧美国产日韩a欧美在线观看| 欧美中文字幕在线视频| 欧美日韩国产区一| 欧美成年人视频网站| 国产麻豆综合| 日韩视频精品在线| 亚洲欧洲精品一区二区三区| 午夜伦欧美伦电影理论片| 99视频在线精品国自产拍免费观看 | 亚洲视频视频在线| 日韩视频一区二区三区在线播放| 亚洲欧美中文另类| 亚洲欧美一级二级三级| 欧美精品久久99久久在免费线| 久久久久久噜噜噜久久久精品| 国产精品分类| 99国产精品| 99精品国产热久久91蜜凸| 久久人人九九| 免费在线日韩av| 国产欧美丝祙| 午夜精品久久久久久久99水蜜桃 | 午夜精品国产| 欧美色区777第一页| 亚洲黄色免费电影| 亚洲精品视频在线看| 久久久久久色| 美女91精品| 国产一区二区三区久久| 亚洲一区www| 亚洲在线观看视频| 欧美黄色免费| 亚洲伦理自拍| 亚洲一区精品电影| 国产精品视频一二三| 亚洲一区三区视频在线观看| 亚洲欧美日韩精品一区二区| 国产精品黄页免费高清在线观看| 99精品视频一区| 亚洲欧美视频在线观看| 国产欧美亚洲精品| 一本色道**综合亚洲精品蜜桃冫| 鲁大师成人一区二区三区| 日韩性生活视频| 欧美日韩国产三区| 亚洲一区久久久| 在线亚洲观看| 国产精品推荐精品| 亚洲综合导航| 久久久久中文| 亚洲激情在线| 欧美网站大全在线观看| 亚洲欧美日韩国产一区二区| 久久久国产精品一区二区中文| 在线观看一区| 欧美日韩国产成人在线| 亚洲图片在线| 麻豆精品精品国产自在97香蕉| 亚洲激情校园春色| 国产精品sss| 久久久久高清| 99这里有精品| 久久久www成人免费毛片麻豆| 亚洲欧洲精品一区二区三区| 国产精品看片资源| 久久综合色综合88| 一区二区三区久久| 久久久久国产精品一区二区| 亚洲精品麻豆| 国产精品成人免费| 看片网站欧美日韩| 亚洲一区成人| 亚洲精品精选| 久久综合伊人77777| 亚洲一区久久久| 91久久综合| 好吊色欧美一区二区三区视频| 欧美日韩国产一区| 免费亚洲电影| 午夜精品短视频| 日韩视频不卡中文| 欧美激情在线播放| 欧美亚洲三级| 99精品欧美一区二区三区| 国产亚洲美州欧州综合国| 欧美日韩黄视频| 欧美成人精品在线播放| 久久精品一区| 欧美亚洲在线观看| 亚洲视频在线观看网站| 亚洲第一天堂无码专区| 久久久久久久久一区二区| 亚洲免费成人av| 一区在线电影| 国产亚洲人成a一在线v站| 欧美日韩国产综合一区二区| 欧美xart系列高清| 麻豆成人小视频| 久久精品亚洲| 久久久久久亚洲精品杨幂换脸 | 亚洲黄色成人| 欧美1区3d| 久久先锋影音| 久久中文字幕一区| 久久久天天操| 久久精品卡一| 久久久久久亚洲精品中文字幕| 欧美亚洲自偷自偷| 亚洲欧美三级伦理| 亚洲视频电影图片偷拍一区| 亚洲精品免费一二三区| 亚洲国产日韩在线一区模特| 1000部精品久久久久久久久| 韩国精品久久久999| 亚洲精品免费一二三区| 午夜精品视频| 午夜精品久久久久久久白皮肤| 在线视频日韩| 亚洲欧美成人精品| 欧美一区高清| 久久久在线视频| 欧美激情精品久久久久久黑人| 亚洲第一精品久久忘忧草社区| 亚洲福利视频二区| 日韩亚洲欧美中文三级| 亚洲一区二区三区精品在线| 亚洲影音先锋| 欧美在线日韩在线| 免费视频一区| 国产精品成人一区二区| 国产精品一区久久| 狠狠色综合播放一区二区| 亚洲国产精品一区制服丝袜 | 国产欧美日韩另类视频免费观看| 国产日韩欧美自拍| 亚洲电影免费观看高清完整版| 亚洲精品中文字幕有码专区| 一区二区欧美日韩| 久久成人国产精品| 欧美大秀在线观看| 亚洲精品免费在线| 欧美亚洲视频| 欧美黄色精品| 狠色狠色综合久久| 宅男在线国产精品| 久久夜色精品国产亚洲aⅴ | 久久天天躁狠狠躁夜夜爽蜜月| 亚洲国产免费| 亚洲女人天堂av| 欧美不卡福利| 国产人成一区二区三区影院| 亚洲国内精品| 久久久五月天| 亚洲午夜高清视频| 欧美粗暴jizz性欧美20| 国产欧美一区二区精品性色 | 国产精品久久夜| 国产一区二区三区不卡在线观看| 亚洲精品护士| 久久综合国产精品台湾中文娱乐网| 亚洲精品字幕| 美女日韩欧美| 国产一级一区二区| 亚洲免费一在线| 亚洲精品少妇网址|