在类构造函数中设置std::vector,为元素构造函数设置不同的值



我有一个具有非平凡构造函数的类:

class mystream
{
public:
    mystream(size_t buffersize,size_t index) : buffersize(buffersize),index(index){}
    size_t buffersize;
    size_t index;
};

mystream实例有一个唯一的id,它对应于它在管理类的向量中的位置:

class mystreammanager
{
public:
   mystreammanager() : streams(8,1024, /* WHAT TO DO HERE ??? */ )
   {
   }
  std::vector<mystream> streams;
};

如何构造vector并将其元素初始化为下标升序值?

一种简洁、清晰且易于调试的方法是将vector的构造推迟到静态类函数:

class mystreammanager
{
public:
   mystreammanager() : streams{ generate_streams(1024, 8) }
   {
   }
private:
  static std::vector<mystream> generate_streams(size_t buffersize, size_t qty)
  {
    std::vector<mystream> result;
    result.reserve(qty);
    for(size_t i = 0 ; i < qty ; ++qty) {
      result.emplace_back(buffersize, i); 
    }
    return result;
  } 
  std::vector<mystream> streams;
};

这是最优的效率,因为:

  1. RVO导致矢量就地构建
  2. 避免列表初始化意味着没有多余的副本。

使用一个循环:

mystreammanager() {
    streams.reserve(8);
    for (int i = 0; i < 8; ++i) {
        streams.emplace_back(1024, i);
    }
}

编译时版本:)

需要c++14,但肯定可以适应c++11

#include <cstddef>
#include <vector>
#include <utility>
class mystream
{
public:
  mystream(size_t buffersize,size_t index) : buffersize(buffersize),index(index){}
  size_t buffersize;
  size_t index;
};
template<size_t... Indexes>
std::initializer_list<mystream>    mystream_maker_impl(std::index_sequence<Indexes...>)
{
  return {{1024, Indexes}...};
}
template<size_t N>
std::initializer_list<mystream>    mystream_maker()
{
  return mystream_maker_impl(std::make_index_sequence<N>());
}
class mystreammanager
{
public:
  mystreammanager() : streams(mystream_maker<8>())
  {
  }
  std::vector<mystream> streams;
};

你可以这样做:

class mystreammanager
{
public:
   mystreammanager() : streams{{1024, 0}, {1024, 1}, {1024, 2}, {1024, 3},
                               {1024, 4}, {1024, 5}, {1024, 6}, {1024, 7}}
   {
   }
  std::vector<mystream> streams;
};

但是循环似乎更安全/更简单。

我使用了@RichardHodges的答案,因为我对我的第一个选择不满意。我制作了这个模板:

template<class T,class ...Args> std::vector<T> generate_with_index(size_t qty,Args ...args)
{
    std::vector<T> result;
    result.reserve(qty);
    for(size_t i = 0 ; i < qty ; ++qty)
        result.emplace_back(i, args...);
    return result;
}

它帮助我避免冗余。从理论上讲,我最喜欢@Drax解决方案,因为它在编译期间做的工作最多。

最新更新