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

标签: c++静态成员静态常量conststatic
546人阅读 评论(0) 收藏 举报
分类:

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

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

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

0
0

查看评论
* 以上用户言论只代表其个人观点,不代表CSDN网站的观点或立场
    个人资料
    • 访问:177463次
    • 积分:3508
    • 等级:
    • 排名:第9417名
    • 原创:167篇
    • 转载:8篇
    • 译文:0篇
    • 评论:65条
    博客专栏
    最新评论