Art*_*hur 9 c++ algorithm predicate std pass-by-reference
我试图从a std::list中删除元素并保留一些已删除元素的统计信息.
为了做到这一点,我使用列表中的remove_if函数,我有一个谓词.我想用这个谓词来收集统计数据.以下是谓词的代码:
class TestPredicate
{
private:
int limit_;
public:
int sum;
int count;
TestPredicate(int limit) : limit_(limit), sum(0), count(0) {}
bool operator() (int value)
{
if (value >= limit_)
{
sum += value;
++count; // Part where I gather the stats
return true;
}
else
return false;
}
};
Run Code Online (Sandbox Code Playgroud)
以下是算法的代码:
std::list < int > container;
container.push_back(11);
TestPredicate pred(10);
container.remove_if(pred)
assert(pred.count == 1);
Run Code Online (Sandbox Code Playgroud)
不幸的是,断言是错误的,因为谓词是按值传递的.有没有办法强制它通过引用传递?
Ker*_* SB 15
传递参考包装器,可从<functional>以下位置获得:
container.remove_if(std::ref(pred));
Run Code Online (Sandbox Code Playgroud)
如果你只有C++ 98/03,但你的编译器有TR1,你可以使用<tr1/functional>,std::tr1::ref如果你对你的谓词做了一个小修改:
#include <tr1/functional>
class TestPredicate : public std::unary_function<int, bool>
{ //^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^
// ...
}
container.remove_if(std::tr1::ref(pred));
Run Code Online (Sandbox Code Playgroud)
如果所有其他方法都失败了,那么您可以相对轻松地破解手动解决方案:
struct predref
{
TestPredicate & p;
bool operator()(int n) { return p(n); }
predref(TestPredicate & r) : p(r) { }
};
container.remove_if(predref(pred));
Run Code Online (Sandbox Code Playgroud)