1

我正在尝试过滤掉带有类型的列表,但这似乎不起作用。我确定我在这里做错了,这是我为重现它而创建的测试:

#include <iostream>

#include <boost/hana.hpp>
#include <boost/hana/ext/std/tuple.hpp>

struct X {
};
struct Y {
};
struct Z {
};

int main(int argc, char **argv) {

    namespace hana = boost::hana;

    constexpr std::tuple<X, Y, Z> list;
    constexpr std::tuple<X> filterlist;

    auto t = hana::filter(list, [&](auto t) {
            return hana::not_(hana::contains(filterlist, hana::decltype_(t)));
        });

    std::cout << "filtered list contains " << hana::size(t) << " items, expected 2 items" << std::endl;

    return 0;
}

基本上我想要的:我有一个类型列表,我想返回一个列表,其中包含不在过滤器列表中的项目。所以在这种情况下,它应该是std::tuple<Y, Z>

该程序的当前输出为:过滤后的列表包含 3 项,预期为 2 项

问候, 马蒂斯

4

1 回答 1

3

问题是您正在检查类型 ( decltype_(X{}) == type<X>{}) 是否在过滤器列表中,其中包含实际对象,而不是类型。换句话说,这有点像您试图将std::type_info表示某种类型T的对象与实际类型的对象进行比较T;这在语义上没有意义。相反,您想要的是以下内容:

#include <iostream>
#include <boost/hana.hpp>
#include <boost/hana/ext/std/tuple.hpp>
namespace hana = boost::hana;

struct X { };
struct Y { };
struct Z { };

int main(int argc, char **argv) {
  constexpr std::tuple<X, Y, Z> list;
  constexpr std::tuple<hana::type<X>> filterlist;
  auto t = hana::remove_if(list, [&](auto t) {
    return hana::contains(filterlist, hana::decltype_(t));
  });

  std::cout << "filtered list contains " << hana::size(t) << " items, expected 2 items" << std::endl;
}

话虽如此,如果您已经有一个filterlist用于其他目的的元组,您仍然可以使用它来过滤:

#include <iostream>
#include <boost/hana.hpp>
#include <boost/hana/ext/std/tuple.hpp>
namespace hana = boost::hana;

struct X { };
struct Y { };
struct Z { };

int main(int argc, char **argv) {
  constexpr std::tuple<X, Y, Z> list;
  constexpr std::tuple<X> filterlist;
  auto t = hana::remove_if(list, [&](auto t) {
    return hana::any_of(filterlist, [&](auto u) {
        return hana::decltype_(u) == hana::decltype_(t);
    });
  });

  std::cout << "filtered list contains " << hana::size(t) << " items, expected 2 items" << std::endl;
}

最后一点,请注意这些构造,因为它们是 O(n^2) 编译时间。如果您需要有效hana::set的查找,请考虑使用(现在实现很糟糕,但当我有更多时间时会变得更好)。

于 2016-09-13T02:52:50.570 回答