我需要定义一些字符串和用这些字符串初始化的数组,这些字符串可供不同的软件使用。我想在这样的头文件中定义它们:
//.h file
const char *serviceStateKindNormal = "Normal";
const char *serviceStateKindUnmanned = "Unmanned";
const char *serviceStateKindScheduledMaintenanceDown = "ScheduledMaintenance (down)";
const char *serviceStateKindScheduledMaintenanceAvailable = "ScheduledMaintenance (available)";
const char *serviceStateKindMajorIncidentInProgress = "MajorIncidentInProgress";
const char *serviceStateKindPartialService = "PartialService";
const char *serviceStateKindOverloaded = "Overloaded";
const char *serviceStateKindGoingDown = "GoingDown";
const char *serviceStateKindDown = "Down";
const char *serviceStateKind[9] = {
serviceStateKindNormal,
serviceStateKindUnmanned,
serviceStateKindScheduledMaintenanceDown,
serviceStateKindScheduledMaintenanceAvailable,
serviceStateKindMajorIncidentInProgress,
serviceStateKindPartialService,
serviceStateKindOverloaded,
serviceStateKindGoingDown,
serviceStateKindDown
};
但是编译器显示
error: initializer element is not constant
serviceStateKindNormal
这里的问题到底是什么?我必须做出什么选择来定义我的变量?
从初始化开始,在文件范围内声明的变量的初始化中放入的所有内容都必须是常量表达式或字符串文字。有一个列表什么是常量表达式,变量值不在其中。因此:
// there is an array of char[2]{'a', '} somewhere in the memory
// we store the pointer value to that array inside the variable a
static const char *a = "a";
// now we want to statically initialize variable b with the same value
static const char *b = a;
将不起作用,因为b
是用指针a
值初始化的,该值不是常量表达式。初始化具有静态存储持续时间的变量时,需要一个常量表达式。
你能做什么?以下内容:
好方法:为什么要存储指向字符串文本
"Normal"
的指针?为什么不将数据本身存储在数组中?令人高兴的是,变量地址是一个常量表达式,所以我们可以在初始化中使用它!请注意,(几乎)所有用例和语义都保持不变,除了一些角落用法(如sizeof(serviceStateKindNormal)
运算符结果)。const char serviceStateKindNormal[] = "Normal"; const char serviceStateKindUnmanned[] = "Unmanned"; const char *serviceStateKind[] = { serviceStateKindNormal, serviceStateKindUnmanned, };
一种奇怪的方法:将指向字符串的指针存储在数组中。由于变量地址是常量表达式,因此这将起作用。请注意,这使
serviceStateKind
成为一个三星变量。使用时需要双重取消引用数组。serviceStateKind
是指向字符串的指针的指针的数组。请注意,如何使用这样的数组非常令人困惑,所以我建议您使用一个结构。const char *serviceStateKindNormal = "Normal"; const char *serviceStateKindUnmanned = "Unmanned"; const char **serviceStateKind[] = { &serviceStateKindNormal, &serviceStateKindUnmanned, }; int main() { // question which one is correct? printf("%sn", *serviceStateKind[1]); printf("%sn", (*serviceStateKind)[1]); printf("%sn", serviceStateKind[0][1]); }
但由于我不认为自己是三星级程序员,我会尽量做到至少两颗星:
const char *serviceStateKindNormal = "Normal";
const char *serviceStateKindUnmanned = "Unmanned";
struct {
// points to a string
const char **pnt;
// an array of pointers to string
} serviceStateKind[] = {
&serviceStateKindNormal,
&serviceStateKindUnmanned,
};
int main() {
// more verbose
printf("%sn", *serviceStateKind[0].pnt);
}
老式的方式-宏。不要这样做。糟糕的是,你可以滥用字符串文字的预处理器连接,将它们连接在一起。
#define serviceStateKindNormal "Normal" #define serviceStateKindUnmanned "Unmanned" const char *serviceStateKind[] = { serviceStateKindNormal, serviceStateKindUnmanned, }; int main() { # magic printf(serviceStateKindNormal " " serviceStateKindUnmanned "n"); }
我想在这样的头文件中定义它们:
如果在头文件中定义变量,则需要用static
标记它们,这样在链接使用该.h
文件的不同.c
文件时就不会出现重复的符号错误。另外,将变量标记为const
也很好,这是对其他程序员的一个提示,表明他们在某种程度上是不可更改的。
在C语言中,常量是指文字常量,如(3.14
、2.718
等)。Const限定对象(任何类型)在C语言术语中都不是常量。
要在C语言术语中创建常量,请使用#define
指令,例如:
#define ServiceStateKindNormal "Normal"
等等。