时间:2021-05-20
C++ 静态成员的类内初始化详解及实例代码
一般来说,关于C++类静态成员的初始化,并不会让人感到难以理解,但是提到C++ 静态成员的"类内初始化"那就容易迷糊了。
我们来看如下代码:
//example.h#include<iostream>#include<vector>using namespace std;class Example{public: static double rate = 6.5; static const int vecSize = 20; static vector<double> vec(vecSize);};//example.cpp#include "example.h"double Example::rate;vector<double> Example::vec;我们需要判断上面的静态数据成员的声明和定义有没有错误,并解释原因。
首先,要谨记:通常情况下,不应该在类内部初始化成员,无论是否为静态成员。
其次,若一定要在类内初始化静态成员,那么就必须满足如下条件才行:
1) 静态成员必须为字面值常量类型的constexpr。
所谓的字面值类型就是通常遇到的:算术类型,引用,指针等。字面值常量类型就是const型的算术类型,引用,指针等。
所谓的constexpr,就是常量表达式,指值不会改变且在编译过程中就能得到计算结果的表达式。比如字面值,或者用常量表达式初始化的const对象也是常量表达式。为了帮助用户检查自己声明/定义的变量的值是否为一个常量表达式,C++11新规定,允许将变量声明为constexpr类型,以便由编译器来进行验证变量是否为常量表达式。
2)给静态成员提供的初始值,必须为常量表达式
注意:在C++ primer 第五版中说:只能给静态成员提供const 整数类型的类内初始值,且该const整数类型的初始值必须是常量表达式。我觉得是有误的!详情见后面分析。
有了这两条原则,我们就可以对上面的代码进行验证了。
1)static double rate = 6.5;
显然不满足第一条:因为rate不是常量类型。改成constexprt static const double rate = 6.5即可
从这里也可以看出初始值不一定必须为const 整数类型。
ps: 如果我们不再这里加入constexprt修饰符的话,编译器会提示错误:error: ‘constexpr' needed for in-class initialization of static data member ‘const double Example::rate' of non-integral type [-fpermissive]
大体意思就是,对于非const整数类型的初始值,如果它是常量表达式的话,我们需要手工在前面添加修饰符constexprt。
至于Example.cpp文件中的定义部分,由于我们已经在类内部进行了初始化,就不需要再在类外部进行定义了。如果非要定义的话,必须采用如下格式:
//example.cppconstexpr const double Example::rate; //其中的const是可以删除的,因为constexprt本身就包含了const2)static const int vecSize = 20;
vecSize是const int类型的,且为常量表达式——满足第一条;提供的初始值为20,是一个常量表达式——满足第二条!且由于是const int型的,前面可以不用修饰符constexpr。
3)static vector<double> vec(vecSize);
错误!vector是模板不是字面值常量类型,所以不满足第一条。应该改为 static vector<double> vec; //仅仅且只能进行声明,不能定义
然后在Example.cpp中进行定义:
static vector<double> vec(Example::vecSize);
现在我们可以在Example.cpp中添加测试代码进行测试了:
#include "example.h"vector<double> Example::vec(Example::vecSize);constexpr const double Example::rate;int main(){ Example::vec.push_back(10.5); cout << Example::vec.back() << endl; cout << Example::rate << endl; cout << Example::vecSize << endl;}执行结果:
wanchouchou@wanchouchou-virtual-machine:~/c++/7.5$ ./Example 10.56.520感谢阅读,希望能帮助到大家,谢谢大家对本站的支持!
声明:本页内容来源网络,仅供用户参考;我单位不保证亦不表示资料全面及准确无误,也不保证亦不表示这些资料为最新信息,如因任何原因,本网内容或者用户因倚赖本网内容造成任何损失或损害,我单位将不会负任何法律责任。如涉及版权问题,请提交至online#300.cn邮箱联系删除。
这里直接给出C#类成员一般初始化顺序:子类静态字段子类静态构造子类实例字段父类静态字段父类静态构造父类实例字段父类实例构造子类实例构造为什么说是“一般”初始化顺
类定义时的静态成员只是声明,静态成员的定义和初始化要在类之外完成C++的static关键字可修饰类成员变量/方法,表示变量/方法不从属于特定对象,而是属于类的。
在以下三种情况下需要使用初始化成员列表:一,需要初始化的数据成员是对象的情况;二,需要初始化const修饰的类成员;三,需要初始化引用成员数据;原因:C++可以
Java中初始化块详解在Java中,有两种初始化块:静态初始化块和非静态初始化块.静态初始化块:使用static定义,当类装载到系统时执行一次.若在静态初始化块
一、类的初始化对于类的初始化:类的初始化一般只初始化一次,类的初始化主要是初始化静态成员变量。类的编译决定了类的初始化过程。编译器生成的class文件主要对定义