C ++初始化非静态成员数组

时间:2011-04-13 03:01:58

标签: c++ arrays default-constructor

我正在编辑一些使用如此定义的全局数组的旧C ++代码:

int posLShd[5] = {250, 330, 512, 600, 680};
int posLArm[5] = {760, 635, 512, 320, 265};
int posRShd[5] = {765, 610, 512, 440, 380};
int posRArm[5] = {260, 385, 512, 690, 750};
int posNeck[5] = {615, 565, 512, 465, 415};
int posHead[5] = {655, 565, 512, 420, 370};

我想让所有这些数组成为下面定义的Robot类的私有成员。但是,当我声明数据成员时,C ++编译器不允许我初始化数据成员。

class Robot
{
   private:
       int posLShd[5];
       int posLArm[5];
       int posRShd[5];
       int posRArm[5];
       int posNeck[5];
       int posHead[5];
   public:
       Robot();
       ~Robot();
};

Robot::Robot()
{
   // initialize arrays
}

我想在Robot()构造函数中初始化这六个数组的元素。除了逐个分配每个元素之外,还有什么方法可以做到这一点吗?

9 个答案:

答案 0 :(得分:15)

如果您的要求确实允许,那么您可以将这5个数组作为类的static数据成员,并在定义.cpp文件时初始化它们,如下所示:

class Robot
{
  static int posLShd[5];
  //...
};
int Robot::posLShd[5] = {250, 330, 512, 600, 680}; // in .cpp file

如果无法做到这一点,请像往常一样使用不同的名称声明此数组,并对构造函数中的数据成员使用memcpy()

修改: 对于非静态成员,可以使用template样式以下(适用于int之类的任何类型)。要更改大小,请同样重载元素数量:

template<size_t SIZE, typename T, T _0, T _1, T _2, T _3, T _4>
struct Array
{
  Array (T (&a)[SIZE])
  {
    a[0] = _0;
    a[1] = _1;
    a[2] = _2;
    a[3] = _3;
    a[4] = _4;
  }
};

struct Robot
{
  int posLShd[5];
  int posLArm[5];
  Robot()
  {
    Array<5,int,250,330,512,600,680> o1(posLShd);
    Array<5,int,760,635,512,320,265> o2(posLArm);
  }
};

C ++ 11

数组初始化现在变得微不足道了:

class Robot
{
   private:
       int posLShd[5];
       ...
   public:
       Robot() : posLShd{0, 1, 2, 3, 4}, ...
       {}
};

答案 1 :(得分:12)

您可以将其设置为静态,或使用C ++ 0x中引入的新初始化

class Robot
{
private:
  int posLShd[5];
  static int posLArm[5];
  // ...
public:
  Robot() :
    posLShd{250, 330, 512, 600, 680} // only C++0x                                                                                     
  {}

  ~Robot();
};

int Robot::posLArm[5] = {760, 635, 512, 320, 265};

答案 2 :(得分:4)

要将另一种方法引入混合(并且不会告诉您按照大多数其他答案做出数组数据成员static - 我假设知道它们是否应该是static),这是我使用的零开销方法:制作static成员函数并让它们返回std::array<>(或{{ 3}}如果您的编译器太旧而无法实现std::std::tr1::

class Robot
{
    static std::array<int, 5> posLShd_impl() { std::array<int, 5> x = {{ 250, 330, 512, 600, 680 }}; return x; }
    static std::array<int, 5> posLArm_impl() { std::array<int, 5> x = {{ 760, 635, 512, 320, 265 }}; return x; }
    static std::array<int, 5> posRShd_impl() { std::array<int, 5> x = {{ 765, 610, 512, 440, 380 }}; return x; }
    static std::array<int, 5> posRArm_impl() { std::array<int, 5> x = {{ 260, 385, 512, 690, 750 }}; return x; }
    static std::array<int, 5> posNeck_impl() { std::array<int, 5> x = {{ 615, 565, 512, 465, 415 }}; return x; }
    static std::array<int, 5> posHead_impl() { std::array<int, 5> x = {{ 655, 565, 512, 420, 370 }}; return x; }

    std::array<int, 5> posLShd;
    std::array<int, 5> posLArm;
    std::array<int, 5> posRShd;
    std::array<int, 5> posRArm;
    std::array<int, 5> posNeck;
    std::array<int, 5> posHead;
public:
    Robot();
};

Robot::Robot()
  : posLShd(posLShd_impl()),
    posLArm(posLArm_impl()),
    posRAhd(posRAhd_impl()),
    posRArm(posRArm_impl()),
    posNeck(posNeck_impl()),
    posHead(posHead_impl())
{ }

答案 3 :(得分:3)

  

除了逐个分配每个元素之外,还有什么方法可以做到这一点吗?

如果您希望使用某些默认值填充数组的所有元素,可以使用std::fill

#include <algorithm>

// ...
Robot::Robot()
{
    std::fill(posLShd, posLShd+5, 13 ) ; // 13 as the default value

    // Similarly work on with other arrays too.
}

如果数组的每个元素都需要填充不同的值,那么在每个索引处分配值是唯一的选择。

答案 4 :(得分:1)

将全局变量保留在代码中,然后使用memcpy()初始化本地数组,将全局数组的内容复制到本地数组。

答案 5 :(得分:1)

这仅与当前问题略相关,但是是一种特殊情况,可以完全解决duplicate

零初始化是C ++语言数组的一种特殊情况。如果初始化列表比数组短,则其余元素初始化为零。例如,重复问题的要求是将类的所有成员初始化为零,包括构造函数中最后一个数组的所有元素:

class myprogram {
public:
    myprogram ();
private:
    double aa,bb,cc;
    double G_[2000];
};

这足以满足构造函数的定义:

myprogram::myprogram():aa(0.0),bb(0.0),cc(0.0), G_{0.} {}

因为G_的第一个元素被显式初始化为值0.,而所有其他元素都被初始化为零。

答案 6 :(得分:0)

// class definition with incomplete static member could be in a header file
Class Robot {
    static const int posLShd[5];
....
// this needs to be placed in a single translation unit only
const int Robot::posLShd[5] = {250, 330, 512, 600, 680};

答案 7 :(得分:0)

不是真的,虽然我同意stefaanv的评论 - 如果它们之前是全球性的,那么将它们变为静态就会让你“轻松分配”,而且它们看起来好像一般都是静态的。

如果这些值偶尔会发生变化,您可以考虑在创建类时从外部文件中读取它们,这样就可以避免重新编译。

您也可以考虑使用std :: vector代替固定数组来提供它提供的一些功能。

答案 8 :(得分:0)

我在这里错过了什么吗?下面的代码有效。只需声明成员,然后立即初始化。

#include <iostream>

class Robot {
  public:
  int posLShd[5] = {250, 330, 512, 600, 680};
  int posLArm[5] = {760, 635, 512, 320, 265};
  int posRShd[5] = {765, 610, 512, 440, 380};
  int posRArm[5] = {260, 385, 512, 690, 750};
  int posNeck[5] = {615, 565, 512, 465, 415};
  int posHead[5] = {655, 565, 512, 420, 370};
  public:
    Robot() {}
    ~Robot() {}
};

int main () {
  Robot obj;
  for (int i = 0;i < 5;i++) {
    std::cout << obj.posRArm[i] << std::endl;
  }
}