C++ zip可变参数模板

use*_*ser 8 c++ iterator tuples variadic-templates c++11

这是C++中一个简单的双容器zip函数:

template <typename A, typename B>
std::list<std::pair<A, B> > simple_zip(const std::list<A> & lhs,
                                       const std::list<B> & rhs)
{
  std::list<std::pair<A, B> >  result;
  for (std::pair<typename std::list<A>::const_iterator,
                 typename std::list<B>::const_iterator> iter
       =
       std::pair<typename std::list<A>::const_iterator,
                 typename std::list<B>::const_iterator>(lhs.cbegin(),
                                                        rhs.cbegin());
       iter.first != lhs.end() && iter.second != rhs.end();
       ++iter.first, ++iter.second)
  {
    result.push_back( std::pair<A, B>(*iter.first, *iter.second) );
  }
  return result;
}
Run Code Online (Sandbox Code Playgroud)

如何将此扩展到具有可变参数模板的任意数量的容器?

我想general_zip接受tuplelistS(每个列表可以包含不同类型的),并返回listtuple秒.

Joh*_*itb 13

看起来这应该有效

std::list<std::tuple<>> simple_zip() {
  return {};
}

template <typename ...T>
std::list<std::tuple<T...>> simple_zip(std::list<T>... lst)
{
  std::list<std::tuple<T...>>  result;
  for (int i = 0, e = std::min({lst.size()...}); i != e; i++) {
    result.emplace_back(std::move(lst.front())...);
    [](...){} ((lst.pop_front(), 0)...);
  }
  return result;
}
Run Code Online (Sandbox Code Playgroud)

@Potatoswatter有一个好的(IMO)评论说,当列表大小不同时,这可能会复制超过需要的东西,并且只使用迭代器会更好,因为pop_front比实际需要更多.我认为以下"修复"迭代器以更多代码为代价.

template <typename ...T>
std::list<std::tuple<T...>> simple_zip(std::list<T>... lst)
{
  std::list<std::tuple<T...>>  result;
  struct {
    void operator()(std::list<std::tuple<T...>> &t, int c,
             typename std::list<T>::iterator ...it) {
      if(c == 0) return;
      t.emplace_back(std::move(*it++)...);
      (*this)(t, c-1, it...);
    }
  } zip;
  zip(result, std::min({lst.size()...}), lst.begin()...);
  return result;
}

std::list<std::tuple<>> simple_zip() {
  return {};
}
Run Code Online (Sandbox Code Playgroud)

  • `struct {int dummy; } pops [] = {(lst.pop_front(),0)...};`聪明的滥用,我必须记住这一点. (2认同)

Pot*_*ter 5

这是对 Johannes 的第一个答案的增量改进。它避免了虚拟struct并避免复制整个输入列表(尽管这并不重要,除非一个列表比其他列表短)。我还使它在所有容器上通用。

但它需要一个样板包索引生成器,无论如何它非常有用

template< std::size_t n, typename ... acc >
struct make_index_tuple {
    typedef typename make_index_tuple<
        n - 1,
        std::integral_constant< std::size_t, n - 1 >, acc ...
    >::type type;
};

template< typename ... acc >
struct make_index_tuple< 0, acc ... >
    { typedef std::tuple< acc ... > type; };
Run Code Online (Sandbox Code Playgroud)

“真正的”实现由一个需要上述实用程序输出的简单函数和一个将包映射到元组的接口函数组成。

template< typename ret_t, std::size_t ... indexes, typename lst_tuple >
ret_t simple_zip( std::tuple< std::integral_constant< std::size_t, indexes > ... >,
    lst_tuple const &lst ) {
    ret_t ret;

    auto iters = std::make_tuple( std::get< indexes >( lst ).begin() ... );
    auto ends = std::make_tuple( std::get< indexes >( lst ).end() ... );

    while ( std::max< bool >({ std::get< indexes >( iters )
                            == std::get< indexes >( ends ) ... }) == false ) {
        ret.emplace_back( * std::get< indexes >( iters ) ++ ... );
    }
    return ret;
}

template< typename ... T >
std::list< std::tuple< typename T::value_type ... > >
simple_zip( T const & ... lst ) {
    return simple_zip
        < std::list< std::tuple< typename T::value_type ... > > > (
        typename make_index_tuple< sizeof ... lst >::type(),
        std::tie( lst ... )
    );
}
Run Code Online (Sandbox Code Playgroud)

至少,这让约翰内斯看起来很容易。这就是大多数可变参数模板算法的样子,因为没有 就无法存储类型可变参数状态tuple,并且没有办法在没有索引包或元递归函数的情况下处理可变参数元组。(编辑:啊,现在 Johannes 使用尾递归本地函数来定义本地包来完成所有这些工作而根本不需要tuple。太棒了……如果你能处理所有的函数式编程;v)。)

  • 好代码。" std::max&lt; bool &gt; ({ std::get&lt; index &gt;( iters )== std::get&lt; index &gt;( ends ) ... }) == false 哈哈,真聪明。 (2认同)