首页
学习
活动
专区
圈层
工具
发布
社区首页 >问答首页 >函数来比较变量的内容,但无法编译。

函数来比较变量的内容,但无法编译。
EN

Stack Overflow用户
提问于 2018-07-20 08:41:24
回答 2查看 185关注 0票数 2

在我的项目中,我详尽地使用了boost-variant。因此,对于我的单元测试,我需要对照特定的T和特定的内容t检查变体的内容。

为此,我设计了函数cmpVariant,以消除单元测试中的杂乱。

在某些情况下,T类型没有配备operator==,因此用户可以传递满足EqualityCompare需求(req/均衡化比较)的函数。

现在,由于某种原因,下面的代码无法编译。它说,没有匹配的功能?

Clang6.0.1编译错误

代码语言:javascript
复制
prog.cc:22:5: error: no matching function for call to 'cmpVariant'
    cmpVariant(number, 3.2, lambdaEquiv); // Fails!
    ^~~~~~~~~~
prog.cc:6:6: note: candidate template ignored: could not match 'function<bool (const type-parameter-0-1 &, const type-parameter-0-1 &)>' against '(lambda at prog.cc:19:24)'
bool cmpVariant(
     ^
1 error generated.

有人知道为什么吗?

代码语言:javascript
复制
#include <iostream>
#include <boost/variant.hpp>
#include <functional>

template<typename V, typename T>
bool cmpVariant(
    const V& variant,
    const T& t,
    const std::function<bool(const T& u, const T& v)>& equiv = [](const T& u, const T& v) {return u == v; })
{
    if (variant.type() != typeid(t)) return false;
    auto v = boost::get<T>(variant);
    return equiv(v, t);
}

int main(int, char**) {
    boost::variant<double, int> number{ 3.2 };
    cmpVariant(number, 3.2);
    auto lambdaEquiv = [](const double& x, const double& y) { return x == y; };
    std::function<bool(const double&, const double&)> equiv = lambdaEquiv;
    cmpVariant(number, 3.2, equiv); // Works!
    cmpVariant(number, 3.2, lambdaEquiv); // Fails!
}
EN

回答 2

Stack Overflow用户

回答已采纳

发布于 2018-07-20 08:52:08

编译器无法将lambda与函数参数类型匹配。您可以通过显式实例化函数调用来修复这个问题:

代码语言:javascript
复制
cmpVariant<boost::variant<double, int>, double>(number, 3.2, equiv);

这显然有点冗长,因此这里有另一种可能将您的函数声明更改为

代码语言:javascript
复制
template<typename V, typename T, typename Fct = std::function<bool(const T& u, const T& v)>>
bool cmpVariant(
    const V& variant,
    const T& t,
    Fct&& f = [](const T& u, const T& v) {return u == v; })
{ /* Same as before. */ }

可以这样叫它

代码语言:javascript
复制
cmpVariant(number, 3.2, equiv); // Type deduction works now.

@DanielLangr在评论中建议的一个改进是使用std::equal_to

代码语言:javascript
复制
template<typename V, typename T, typename Fct = std::equal_to<T>>
bool cmpVariant(
      const V& variant,
      const T& t,
      Fct&& f = std::equal_to<T>{})
{ /* Again, same as before. */ }

这里的一个优点是摆脱了std::function及其经常不必要的开销。

票数 3
EN

Stack Overflow用户

发布于 2018-07-20 08:58:11

接受比较器参数的方式会导致演绎问题,因此您可能希望将比较器更改为模板参数(可能避免构造沉重的std::function对象):

代码语言:javascript
复制
template<typename T> class t_EquilityComparator
{
    public: bool operator ()(const T& u, const T& v) const { return u == v; }
};

template<typename V, typename T, typename Comparator = t_EquilityComparator<T>>
bool cmpVariant(
    const V& variant,
    const T& t,
    const Comparator & equiv = Comparator{})
{
    if (variant.type() != typeid(t)) return false;
    auto v = boost::get<T>(variant);
    return equiv(v, t);
}

int main(int, char**) {
    boost::variant<double, int> number{ 3.2 };
    cmpVariant(number, 3.2);
    auto equiv = [](const double& x, const double& y) { return x == y; };
    cmpVariant(number, 3.2, equiv); // This line fails to compile! Why?
}

联机编译器

票数 1
EN
页面原文内容由Stack Overflow提供。腾讯云小微IT领域专用引擎提供翻译支持
原文链接:

https://stackoverflow.com/questions/51438506

复制
相关文章

相似问题

领券
问题归档专栏文章快讯文章归档关键词归档开发者手册归档开发者手册 Section 归档