将 std::vector<int> 设置为一个范围

Posted

技术标签:

【中文标题】将 std::vector<int> 设置为一个范围【英文标题】:Set std::vector<int> to a range 【发布时间】:2012-08-11 12:54:01 【问题描述】:

std::vector&lt;int&gt; 设置为范围的最佳方法是什么,例如3 到 16 之间的所有数字?

【问题讨论】:

这可能会有所帮助:***.com/a/7256008/8331 【参考方案1】:

如果您支持 C++11 或正在使用 the STL,则可以使用 std::iota

std::vector<int> v(14);
std::iota(v.begin(), v.end(), 3);

如果没有,也可以自己实现。

如果你可以使用boost,那么一个不错的选择是boost::irange

std::vector<int> v;
boost::push_back(v, boost::irange(3, 17));

【讨论】:

v.reserve(14) 将保存默认初始化。 @rhalbersma 我不确定这是否可行。这只是在必要时更改向量的内部存储,但 iota 需要一个有效的迭代器范围。 如果您使用std::back_inserter则不会 @rhalbersma 然后iota 什么时候停止?没有办法告诉它“在 N 个数字后停止”。 您的代码很好,但我更喜欢我自己的 iota_n 答案来进行预先内存预留,而不是默认初始化 0...0 然后立即用 3...16 覆盖.如果 N = 140 亿而不是 14 会怎样?【参考方案2】:
std::vector<int> myVec;
for( int i = 3; i <= 16; i++ )
    myVec.push_back( i );

【讨论】:

KISS +1。你的也是唯一的答案(到目前为止),你可以清楚地看到代码中的范围(从 3 到 16 循环)。其他都使用 3(范围开始)和 14(元素数量)。 在这里做一个myVec.reserve(14) 会更好。 啊,但是对于real C++来说这有点太简单了。您的解决方案长度少于 20 行,并且不会触发 30 个带有拼写错误的神秘模板错误 :)。不过,它会执行不需要的初始化,但我怀疑很少会感觉到差异......【参考方案3】:

参见例如这个question

#include <algorithm>
#include <iostream>
#include <iterator>
#include <vector>

template<class OutputIterator, class Size, class Assignable>
void iota_n(OutputIterator first, Size n, Assignable value)

        std::generate_n(first, n, [&value]() 
                return value++;
        );


int main()

    std::vector<int> v;                   // no default init
    v.reserve(14);                        // allocate 14 ints
    iota_n(std::back_inserter(v), 14, 3); // fill them with 3...16

    std::for_each(v.begin(), v.end(), [](int const& elem) 
        std::cout << elem << "\n";
    );
    return 0;

Ideone 上的输出

【讨论】:

有了back_inserter_iterator,你就不需要调用reserve了,不是吗? @jrok 你不需要,但它更有效。 back_inserter 会调用push_back,但如果你插入大量元素,push_back 会依次进行大量的重新分配。 @jrok 你永远不需要打电话给reserve。它只用于避免重新分配。【参考方案4】:

std::iota - 很有用,但它需要迭代器,在创建向量之前,......所以我采取了自己的解决方案。

#include <iostream>
#include <vector>

template<int ... > struct seq typedef seq type;;

template< typename I, typename J> struct add;
template< int...I, int ...J>
struct add< seq<I...>, seq<J...> > : seq<I..., (J+sizeof...(I)) ... >;


template< int N>
struct make_seq : add< typename make_seq<N/2>::type, 
                       typename make_seq<N-N/2>::type > ;

template<> struct make_seq<0> typedef seq<> type; ;
template<> struct make_seq<1> typedef seq<0> type; ;


template<int start, int step , int ... I>
std::initializer_list<int> range_impl(seq<I... > )

    return  (start + I*step) ...;


template<int start, int finish, int step = 1>
std::initializer_list<int> range()
 
    return range_impl<start, step>(typename make_seq< 1+ (finish - start )/step >::type  ); 


int main()

    std::vector<int> vrange  range<3, 16>( ) ;

    for(auto x : vrange)std::cout << x << ' ';




Output:

  3 4 5 6 7 8 9 10 11 12 13 14 15 16

【讨论】:

这个例子的神秘部分将放在一个你不会经常看到的.h文件中,如果有的话。我认为使用range&lt;3, 16&gt;() 比使用std::iota 更好。【参考方案5】:

尝试使用std::generate。它可以根据公式为容器生成值

std::vector<int> v(size);
std::generate(v.begin(),v.end(),[n=0]()mutablereturn n++;);

【讨论】:

以上是关于将 std::vector<int> 设置为一个范围的主要内容,如果未能解决你的问题,请参考以下文章

如何将 std::vector<thrust::device_vector<int>> 转换为 int**?

将 std::vector<int> 设置为一个范围

将 std::vector<int> 的每个值重置为 0 的最快方法

将 std::stack 复制到 std::vector

将 std::vector<int> 从原始内存转换为数组[重复]

我可以制作一个线程安全的 std::atomic<vector<int>> 吗?