[英]Adapting Map Iterators Using STL/Boost/Lambdas
請考慮以下非工作代碼:
typedef map<int, unsigned> mymap;
mymap m;
for( int i = 1; i < 5; ++i )
m[i] = i;
// 'remove' all elements from map where .second < 3
remove_if(m.begin(), m.end(), bind2nd(less<int>(), 3));
我正在嘗試從這個地圖中刪除.second < 3
元素。 這顯然寫得不正確。 如何使用以下方法正確編寫:
bind
+ less<>
標准STL函數對象和技術,但無需編寫自定義函子 我知道我不會erase
元素。 別擔心; 我只是簡化要解決的問題。
我不確定如何僅使用STL綁定器來執行此操作,但我認為您的主要問題是傳遞給您要remove
的仿函數的不僅僅是一個int
而是一pair<int, unsigned>
。
使用boost :: bind你會這樣做:
remove_if(m.begin(), m.end(), bind(&std::pair<int, unsigned>::second, _1) < 3);
使用lambda函數,它是這樣的:
remove_if(m.begin(), m.end(), [](const std::pair<int, unsigned>& p) { return p.second < 3; } );
我沒有檢查過這個編譯,對不起。
remove_if
不適用於關聯容器。 但remove_copy_if
可能會起作用,但代價是復制地圖。 相反,我會用count_if
來做。
1)使用bind + less <>的標准STL函數對象和技術,但不必編寫自定義函子
// I don't think it can be done with standard c++ without introducing new functors and adaptors.
std::size_t count = std::count_if( m.begin(), m.end(),
std::sgi::compose1(
std::bind_2nd( std::less<int>(), 3 ),
&boost::get<1,mymap::value_type> ) );
2)Boost.Bind
std::size_t count = std::count_if( m.begin(), m.end(),
boost::compose_f_gx(
&boost::bind( std::less<int>, _1, 3 )
&boost::get<1,mymap::value_type> ) );
3)C ++ 0x Lambdas
std::size_t count = std::count_if( m.begin(), m.end(),
[]( const mymap::value_type& item )
{ return item.second < 3; } );
如果你真的想要remove_if行為,你需要推出自己的算法。 我不相信有任何修改標准算法可以使用關聯容器。
template< typename FwdIter, typename AssocCont, typename Pred >
std::size_t assoc_remove_if( FwdIter iter, FwdIter end, AssocCont& cont, Pred pred )
{
std::size_t count = 0;
while( iter != end )
{
if( pred(*iter) )
{
++count;
iter = cont.erase(iter);
}
else
{
++iter;
}
}
return count;
}
雖然由於上面提到的原因我無法使remove_if
算法工作,但我得到了count_if
算法來處理一些精心設計的函子定義和組合。 這些沒有在標准中定義,但它們受到SGI STL可用的啟發。
template <class Pair>
struct select2nd : std::unary_function<Pair, typename Pair::second_type>
{
typedef std::unary_function<Pair, typename Pair::second_type> super;
typedef typename super::result_type result_type;
typedef typename super::argument_type argument_type;
result_type & operator ()(argument_type & p) const {
return p.second;
}
result_type const & operator ()(argument_type const & p) const {
return p.second;
}
};
template <class UnaryFunc1, class UnaryFunc2>
struct unary_compose : std::unary_function<typename UnaryFunc2::argument_type,
typename UnaryFunc1::result_type>
{
typedef std::unary_function<typename UnaryFunc2::argument_type,
typename UnaryFunc1::result_type> super;
typedef typename super::result_type result_type;
typedef typename super::argument_type argument_type;
UnaryFunc1 func1_;
UnaryFunc2 func2_;
unary_compose(UnaryFunc1 f1, UnaryFunc2 f2) : func1_(f1), func2_(f2) {}
result_type operator () (argument_type arg) {
return func1_(func2_(arg));
}
};
template <class UnaryFunc1, class UnaryFunc2>
unary_compose<UnaryFunc1, UnaryFunc2>
compose1(UnaryFunc1 f1, UnaryFunc2 f2) {
return unary_compose<UnaryFunc1, UnaryFunc2>(f1,f2);
};
int main(void) {
typedef std::map<int, unsigned> mymap;
mymap m;
for(int i = 0; i < 5; ++i )
m[i] = i;
std::cout << "Count = "
<< std::count_if(m.begin(), m.end(),
compose1(std::bind2nd(std::less<int>(), 3), select2nd<mymap::value_type>()))
<< std::endl;
}
聲明:本站的技術帖子網頁,遵循CC BY-SA 4.0協議,如果您需要轉載,請注明本站網址或者原文地址。任何問題請咨詢:yoyou2525@163.com.