今天在静态成员上踩坑不少,特此记录。
c++类的静态成员需要在类内声明,而在类外进行定义,如下
class M
{
public:
static int m;
};
int M::m = 90;
int main()
{
cout<<M::m;
}
类内静态常量可以在类内直接初始化,如下,当然也可用上面的方法
class M
{
public:
const static int m = 9;
};
int main()
{
cout<<M::m;
}
类内静态常量数组的定义
class M
{
public:
const static int m[];//声明
M()
{
cout<<m[2]<<endl;
}
};
const int M::m[8]={0,9,8,8};//定义
int main()
{
M m = M();
}
那么问题来了,可不可以直接定义呢?静态常量数组也是常量啊,当然可以
class M
{
public:
static const int m[] = {2,3,4};
M()
{
cout<<m[2]<<endl;
}
};
int main()
{
M m = M();
}
如果你把上面代码copy或者敲下来编译会发现有个错误:
In-class initializer for static data member of type ‘const int [78]’ requires ‘constexpr’ specifier
大概意思是:const int [ 3 ]的静态数据成员初始化需要constexpr类
为什么呢?
因为const只表明这个变量是的值是不变的,但并没表明该变量需要在编译期确定,巧的是,数组就恰恰需要在编译期确定,而constexpr,则表明这个值不仅是值不变的,而且也是编译期确定的。
那好,我们改
class M
{
public:
static constexpr int m[] = {2,3,4};
M()
{
// cout<<m[2]<<endl; 注意这句注释,往下看
}
};
int main()
{
M m = M();
}
没错,编译是成功的,那么,把上面的注释去掉呢,铛,两个错误来袭。。。
clang: error: linker command failed with exit code 1 (use -v to see invocation)
链接器命令失败,退出码1(使用- v查看调用)
“M::m”, referenced from:
查了好多问答文章,没有找到此错误原因,回过头再看书(c++primer),看到这样一段话:
class A {
static constexpr int p = 30;
double d[p];
};
如果某个静态成员的应用场景仅限于编译器可以替换它的值的情况,则一个初始化的const或constexpr static不需要分别定义。相反,如果我们将它用于值不能替换掉应用场景中,则该成员必须有一条定义语句。
显然,静态常量数组的元素是不能被替换掉的,所以。。。
class M
{
public:
static constexpr int m[] = {2,3,4};
M()
{
cout<<m[2]<<endl;
}
};
constexpr int M::m[];//如果在类的内部提供了一个初始值,
//则成员的定义不能再指定一个初始值了
int main()
{
M m = M();
}
以上,就是今一下午踩坑的收获,刚出坑,脚下难免有泥,若有错误,欢迎批评指正!