c++类的静态成员踩坑记录

本文深入探讨C++中静态成员变量的定义与初始化方法,包括不同类型的静态成员变量及其在类内外的正确使用方式,并解析编译时遇到的具体错误及其解决办法。

摘要生成于 C知道 ,由 DeepSeek-R1 满血版支持, 前往体验 >

今天在静态成员上踩坑不少,特此记录。

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();
}

以上,就是今一下午踩坑的收获,刚出坑,脚下难免有泥,若有错误,欢迎批评指正!

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值