初始化子类中父级的const成员

时间:2013-03-07 12:40:06

标签: c++ oop initialization

我甚至不确定这是可能的,所以我想澄清一下。我有一个带有const数组字符串的父类,我希望它的子类初始化,例如:

class CParent{
  CParent();
  const char** strings;
};

和儿童班

class CChild:CParent{
  CChild();
};

CChild::CChild()
: CParent::strings{
  "First",
  "Second"
}
{
  CParent();
  // some code
}

我需要这个,因为我将调用CParent的构造函数,我需要使用字符串。它可以通过参数传递来完成,但我想知道这样的事情是否可行。


编辑:我在这里重写代码时忘了写一些东西,所以我宁愿复制粘贴它,所以我现在不会忘记任何事情。我使用Andy Prowl的帮助用字符串和向量重写了它:

class CMenu {
public:
    CMenu(std::vector<std::string> const& s);
protected:
    std::vector<std::string> choicesStr;
};

CMenu::CMenu(std::vector<std::string> const & s) : choicesStr(s) {
    // code code
}

class CGameTypeMenu : public CMenu {
public:
    CGameTypeMenu();
};

CGameTypeMenu::CGameTypeMenu() 
 :CMenu(std::vector<std::string>("aaa","bbb")){ // This is where I 
                                                   get some nasty errors

}

错误如下所示:

In file included from /usr/include/c++/4.7/vector:63:0,
                 from CMenu.h:13,
                 from CGameTypeMenu.h:11,
                 from CGameTypeMenu.cpp:8:
/usr/include/c++/4.7/bits/stl_uninitialized.h:77:3:   required from ‘static _ForwardIterator std::__uninitialized_copy<_TrivialValueTypes>::__uninit_copy(_InputIterator, _InputIterator, _ForwardIterator) [with _InputIterator = const char*; _ForwardIterator = std::basic_string<char>*; bool _TrivialValueTypes = false]’
(5+ more similar lines follow)

4 个答案:

答案 0 :(得分:4)

只是以正确的方式执行(此解决方案仅适用于C ++ 11):

#include <vector>
#include <string>

class CParent
{
    protected:
//  ^^^^^^^^^^ Make sure your constructor is at least protected,
//             or it will be inaccessible to derived classes

    CParent(std::vector<std::string> const& s) : strings(s) { };

    std::vector<std::string> strings;
//  ^^^^^^^^^^^^^^^^^^^^^^^^ Use containers and classes from the C++ Standard
//                           Library rather than raw pointers
};

class CChild : public CParent
//           ^^^^^^^^^^^^^^^^ I suppose you forgot this.
//                            Inheritance is private by
//                            default, which does not seem
//                            to be what you want   
{
    CChild();
};

CChild::CChild()
    :
    CParent({"First", "Second"}) // C++11 ONLY!
//          ^^^^^^^^^^^^^^^^^^^
//          Implicit construction of the vector of string
{
    // some code
}

答案 1 :(得分:1)

您可以让CChild的构造函数调用CParent的构造函数并将您的数组作为其参数传递。然后你可能最终会尝试做这样的事情:

CChild() : CParent({ "First", "Second" }) { }

称为大括号初始化,只有C ++ 11支持才有可能。如果您需要C ++ 03解决方案并且希望继续使用const char*来存储字符串,那么:

class CParent {
public:
    CParent(const char* s[]) : strings(s) { }
    const char** strings;
};

class CChild : public CParent {
public:
    CChild() : CParent(strings_) { }
private:
    static const char* strings_[];
};

const char* CChild::strings_[] = { "First", "Second" };

请注意,由于strings_是一个数组,因此必须在源文件中初始化此const成员。

虽然我建议您使用std::vector而不是C风格的数组和std::string而不是C风格的字符串。在这种情况下,此代码可能如下所示:

typedef std::vector<std::string> StringVector;

class CParent {
public:
    CParent(const std::vector<std::string>& s) : strings(s) { }
    const StringVector strings;
};

class CChild : public CParent {
public:
    CChild()
    : CParent(strings_) { }
private:
    static const std::string s_[];
    static const StringVector strings_;
};

const std::string CChild::s_[] = { "First", "Second" };
const StringVector CChild::strings_ =
  StringVector(CChild::s_, CChild::s_ + sizeof(CChild::s_)/sizeof(std::string));

答案 2 :(得分:0)

成员初始化CParent成员的唯一方法是通过适当的CParent构造函数。

作为一种变通方法,如果父成员可访问,即protectedpublic,则可以通过子项构造函数中的赋值初始化父成员。

如果成员真的是const,即

,这种解决办法当然是不可能的
const char ** const strings;

但正确的方法是保持父成员私有并提供适当的父构造函数。

答案 3 :(得分:0)

初始化基类的const成员的唯一方法是将初始化参数传递给派生类构造函数的初始化列表中基类的构造函数,并且基本构造函数接受此参数以初始化const成员他的初始化列表。

“Inside”任何构造函数(传递初始化列表)任何成员(派生或基础)已经构建(默认不显式),并且const不能再更改。