C ++内联初始化静态const char *

时间:2014-07-04 10:37:55

标签: c++

为什么我无法在头文件中初始化静态const char *? 在我的代码中,我有我的类标题:

static const char* xml_ID_TAG;

和cpp:

const char* Class::xml_ID_TAG = "id";

xml_ID_TAG变量包含XML文档的属性字符串。 因为它是静态的,const,原始类型(char *)等...我无法弄清楚为什么编译器禁止编写类似的东西:

static const char* xml_ID_TAG = "id";

我正在使用MSVC2013编译器,为上面的示例提供了错误:“错误:具有类内初始化程序的成员必须是const”

3 个答案:

答案 0 :(得分:14)

一般来说,您必须在一个翻译单元中定义静态成员,并且该语言通过禁止您在周围的类定义中为此类成员编写初始化器来帮助强制执行此操作:

struct T
{
   static int x = 42;
   // ^ error: ISO C++ forbids in-class initialization of
   // non-const static member 'T::x'
};

但是,为了方便起见,对常量进行了特殊的例外处理:

struct T
{
   static const int x = 42;
   // ^ OK
};

请注意,在大多数情况下,您仍然需要定义常量(在.cpp文件中将是最佳位置):

const int T::x;

  

[C++11: 9.4.2/3]:]如果非易失性const static数据成员是整数或枚举类型,则其在类定义中的声明可以指定大括号或等于初始值其中作为赋值表达式的每个 initializer-clause 是一个常量表达式(5.19)。可以使用static说明符在类定义中声明文字类型的constexpr数据成员;如果是这样,它的声明应指定一个大括号或等于初始化,其中作为赋值表达式的每个 initializer-clause 是一个不断表达。 [注意:在这两种情况下,成员可能会出现在常量表达式中。 -end note] 如果程序中的 odr-used (3.2),并且命名空间范围定义不包含初始化程序,则仍应在命名空间作用域中定义该成员


现在,您的会员不是int,即使const char* const也不属于“整体类型”:

struct T
{
   static const char* const str = "hi";
   // ^ error: 'constexpr' needed for in-class initialization of
   // static data member 'const char* const T::str' of non-integral type
};

是“文字类型”的;你的结果是,如果你这样写:

static constexpr const char* const xml_ID_TAG = "id";
//     ^^^^^^^^^             ^^^^^
你应该没问题。无论如何,这可能更有意义:你为什么要改变指针?

答案 1 :(得分:3)

" 因为它是[...] const " - 没有。

您需要const char* const - 假设您正在使用C ++ 11。

否则你必须把定义放在cpp文件中。

请参阅:How to initialize a static const member in C++?

答案 2 :(得分:3)

因为每个文件编译单元都存储了字符串文字(例如"id")。因此,如果它们位于头文件中,则为包含它的每个源文件存储不同的实例。所以你的'初始化'试图为static的每个编译单元在#include变量中存储不同的值..