boost::mem_fn 從成員函數指針到函數對象
C++通過范型編程倒是映入了大量函數編程的思想和手法,但終究強類型的語言,有時代碼就不是那么優雅了.一般情況下函數指針和函數對象(functor)是可以互換的,如std::for_each的第三個參數,但標準庫中有4個函數配接器(not1,not2,bind1st,bind2nd)是不能接受函數指針,必須使用std::ptr_fun將函數指針封裝成函數對象.還有一種情況是函數指針指向的函數的參數類型與函數模板需要的不匹配,如果要通過for_each調用每個Iterator的解引用(dereference)的對象的某個成員函數.直接使用諸如ClassA::*member_func的成員函數指針顯然是想當然的做法,調用的語法都不一致,結果是一大堆編譯錯誤信息.沒有關系可以使用std::mem_fun和std::mem_fun_ref.如for_each(classAVector.begin(),classAVector.end(),mem_fun_ref(ClassA::*member_func)).標準庫中的這一套函數適配器(functor adapters)使用起來要充分考慮使用的場合.不同的情況使用不同配接器.特別是您要區分for_each的容器是Container<T>還是Container<T*>,前者使用std::mem_fun_ref,后者使用std::mem_fun.這一段的詳細討論可一參照Effective STL的第40條和第41條.(個人感覺這幾個配接器的名字真有的...fun).
如果您實在是有點受不了或者你不能確定容器中究竟放的是什么而且你愿意使用一下準標準庫,就可以考慮boost庫的mem_fn.boost::mem_fn也不是很特別的boost成員.使用boost::mem_fn好處:不用關心容器中是T還是T*甚至是boost::share_ptr<T>,另外買一贈一fn比fun更好聽一點.下面抄錄boost的幫助代碼就很說明問題了.
struct X
{
void f();
};
void g(std::vector<X> & v)
{
std::for_each(v.begin(), v.end(), boost::mem_fn(&X::f));
};
void h(std::vector<X *> const & v)
{
std::for_each(v.begin(), v.end(), boost::mem_fn(&X::f));
};
void k(std::vector<boost::shared_ptr<X> > const & v)
{
std::for_each(v.begin(), v.end(), boost::mem_fn(&X::f));
};
{
void f();
};
void g(std::vector<X> & v)
{
std::for_each(v.begin(), v.end(), boost::mem_fn(&X::f));
};
void h(std::vector<X *> const & v)
{
std::for_each(v.begin(), v.end(), boost::mem_fn(&X::f));
};
void k(std::vector<boost::shared_ptr<X> > const & v)
{
std::for_each(v.begin(), v.end(), boost::mem_fn(&X::f));
};
posted on 2007-07-24 12:34 五點半 閱讀(3058) 評論(0) 編輯 收藏 引用 所屬分類: 使用標準庫和Boost庫