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

ABC*_*lus 4 c++

为什么我不能在头文件中初始化静态const char*?在我的代码中,我在我的类头中:

static const char* xml_ID_TAG;
Run Code Online (Sandbox Code Playgroud)

在cpp中:

const char* Class::xml_ID_TAG = "id";
Run Code Online (Sandbox Code Playgroud)

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

static const char* xml_ID_TAG = "id";
Run Code Online (Sandbox Code Playgroud)

我正在使用MSVC2013编译器,给出上面的错误示例:"错误:具有类内初始化程序的成员必须是const"

Lig*_*ica 14

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

struct T
{
   static int x = 42;
   // ^ error: ISO C++ forbids in-class initialization of
   // non-const static member 'T::x'
};
Run Code Online (Sandbox Code Playgroud)

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

struct T
{
   static const int x = 42;
   // ^ OK
};
Run Code Online (Sandbox Code Playgroud)

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

const int T::x;
Run Code Online (Sandbox Code Playgroud)

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


现在,您的会员不是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
};
Run Code Online (Sandbox Code Playgroud)

但它一个"字面型"; 你的结果是,如果你这样写:

static constexpr const char* const xml_ID_TAG = "id";
//     ^^^^^^^^^             ^^^^^
Run Code Online (Sandbox Code Playgroud)

你应该没问题.无论如何,这可能更有意义:你为什么要改变指针?


Pau*_*ans 5

因为字符串文字(例如"id")是按文件编译单元存储的。因此,如果它们位于头文件中,则为包含它的每个源文件存储一个不同的实例。因此,您的“初始化”尝试在static变量中为每个编译单元存储不同的值#include