使用 boost hana 从类型列表中删除重复项的统一方法

uniform way to remove duplicates from a typelist using boost hana

我正在尝试熟悉 boost::hana。作为练习,我想创建一个函数,使用用户提供的比较函数从 hana::tuple 中删除重复项。我面临的问题与使用 hana::type_c 将类型存储为对象有关。这是我的

#include <boost/hana/equal.hpp>
#include <boost/hana/tuple.hpp>
#include <boost/hana/unpack.hpp>
#include <boost/hana/pair.hpp>
#include <boost/hana/any_of.hpp>
#include <boost/hana/second.hpp>
#include <boost/hana/fold.hpp>
#include <boost/hana/core/make.hpp>
#include <boost/hana/core/tag_of.hpp>

#include <iostream>

template <class>
struct what_is;

namespace hana = boost::hana;

// simply push back an element to the sequence
auto push_back = [](auto seq, auto t) {

    using namespace boost::hana;
    return unpack(seq, [&](auto&&... element){return make<typename tag_of<decltype(seq)>::type>(element..., t);});
};

// this is the main function
auto remove_duplicates = [](auto seq, auto comp) {

    using namespace boost::hana;

    auto f = [&](auto state, auto el){
        return if_( any_of(state, partial(comp, el)),
                   [=](){return state;},
                   [=](){return push_back(state, el);})();
    };

    return fold(seq, make<typename tag_of<decltype(seq)>::type>(), f);
};

// user-defined comparison function
// elements are considered equal if only second element of pairs are equal
auto comp_pair = [](auto&& t1, auto&& t2) {

    using namespace boost::hana;
    return equal(second(t1), second(t2));
};


int main() {

    auto my_tuple1 = hana::tuple_t<int, float, double, int, float>;
    auto no_dups1 = remove_duplicates(my_tuple1, hana::equal); // this is fine, decltype(no_dups1) -> tuple< type<int>, type<float>, type<double> >

    auto my_tuple2 = hana::tuple_t< hana::pair<int, int>, hana::pair<float, int>, hana::pair<float, float> >;
//    auto no_dups2 = remove_duplicates(my_tuple2, comp_pair); // what I want here is tuple< type<pair<int, int>>, type<pair<float, float>> >
}

最后一行会产生问题,因为没有第二个元素可以从 hana::type<pair<X,Y>> 中提取。为此,我必须创建一个非常丑陋的序列,例如 tuple< pair<type<int>, type<int>>, pair<type<double>, type<int>>, pair<type<float>, type<double>> >。你可以想象这会很快变得糟糕,例如,如果我有一个序列 tuple<int, pair<X,Y>, double, float> 等等。我有什么办法可以创建一个统一的方法来处理这个问题吗?我来自 MPL/fusion 背景,在那里我可以直接使用类型而不需要包装类型。谢谢

与 Fusion 不同,Hana 不会隐式地将值转换为类型。通常,这很好,因为这意味着您可以使用更具表现力的值语法。另一方面,对于某些您确实想要提取包装类型的用例,您必须使用 Hana 明确地执行此操作,而 Fusion 会在后台为您执行此操作。

我看到了您要实现的目标的两个选项。第一个解决方案是更改 comp_pair 函数,以便它自己解包对:

template <typename T1, typename U1, typename T2, typename U2>
constexpr auto comp_pair(hana::basic_type<hana::pair<T1, U1>>,
                         hana::basic_type<hana::pair<T2, U2>>)
{
    return hana::type_c<U1> == hana::type_c<U2>;
}

...

auto no_dups2 = remove_duplicates(my_tuple2, [](auto pair1, auto pair2) {
    return comp_pair(pair1, pair2);
});

第二个解决方案,我觉得更惯用,是将您的类型实际保存为对象:

// This could arguably be part of Hana, just like we provide tuple_t
template <typename T, typename U>
constexpr auto pair_t = hana::make_pair(hana::type_c<T>, hana::type_c<U>);

auto tuple2 = hana::make_tuple(pair_t<int, int>, pair_t<double, int>, pair_t<float, double>);
auto nodups2 = remove_duplicates(tuple2, [](auto p1, auto p2) {
  return hana::second(p1) == hana::second(p2);
});

然而,你说:

As you can imagine this can grow bad really quickly, for instance if I had a sequence tuple<int, pair<X,Y>, double, float> etc. Is there any way I can create a uniform way to deal with this?

我不确定我是否关注这里。您是说您可能想要 tuple<type<int>, pair<type<X>, type<Y>>, type<double>, type<float>> 之类的东西,并且正在寻找实现此目的的通用方法吗?如果是这样,那么我必须说我非常怀疑有更好的方法来实现您想要做的事情。如果您提供更多背景信息,我可以尽力提供帮助。

希望对您有所帮助!