如何在一元谓词中使用类成员变量

How to use a class member variable in a unary predicate?

本文关键字:变量 成员 谓词 一元      更新时间:2023-10-16

假设我有以下代码:

#include <vector>
#include <algorithm>
#include <cmath>
bool pred(float const &val)
{
    //return fabs(val) < epsilon;
}
class Foo
{
public:
    Foo(float eps) : epsilon(eps) { }
    int Bar(std::vector<float> vec)
    {
        return std::count_if(vec.begin(), vec.end(), pred);
    }
private:
    float epsilon;
}

我想将一元谓词pred()作为类方法,以便它可以访问epsilon成员变量。我见过使用operator()的例子,但如果我做对了,操作符应该比较整个类实例。我只需要比较Bar()中给定的std::vector的元素。是否可以创建一元谓词作为类成员?如果不能,还有其他方法吗?

如果你正在使用c++ 11,你应该是一个简单的方法是lambda

int Bar(std::vector<float> vec)
{
    return std::count_if(vec.begin(), vec.end(),[this](const float &f)
    {
        return fabs(val) < epsilon;
    });
}

c++ 98的解决方案:

class Pred{
    private:
        float eps;
    public:
        Pred(float eps) : eps(eps){
        }
        bool operator()(const float& val){
            return val < eps;
        }
};

并像这样使用:

int Bar(std::vector<float> vec)
{
    Pred pred(epsilon);
    return std::count_if(vec.begin(), vec.end(), pred);
}

请注意,此解决方案适用于任何需要额外数据的谓词;编写一个包含附加数据的类作为类成员,并重载()操作符。您可以使用适当的数据初始化类的对象,然后将其用作谓词。

您的问题是pred()需要两个参数,而std::count_if想要一个只接受一个参数的函数。@Praetorian使用lambda的c++11解决方案是最佳的。但是如果没有这个权限,你应该可以使用std::bind1st,它可以为你完成operator()的解决方案,而不需要你显式地创建一个类来完成它。

#include <vector>
#include <algorithm>
#include <cmath>
#include <functional>
bool pred(float const &val, float epsilon)
{
    return fabs(val) < epsilon;
}
class Foo
{
public:
    Foo(float eps) : epsilon(eps) { }
    int Bar(std::vector<float> vec)
    {
        return std::count_if(vec.begin(), vec.end(), std::bind1st(std::less<float>(), epsilon));
    }
private:
    float epsilon;
};