函数 const 本地对象在C++会发生什么
What happens to the function const local object in C++?
我想知道为什么会这样:下面的两个例子提供了非常不同的结果,第一个实际上让我震惊。
示例 1:
#include <iostream>
class A
{
public:
A() {
for(int i = 0; i < 10000; ++i)
for(int j = 0; j < 10000; ++j)
m += i+j;
}
~A() {}
double m;
};
void foo()
{
const A a;
std::cout << a.m << std::endl;
}//foo
int main()
{
for(int i = 0; i < 10; ++i)
foo();
return 0;
}//main
结果:
9.999e+11
1.9998e+12
2.9997e+12
3.9996e+12
4.9995e+12
5.9994e+12
6.9993e+12
7.9992e+12
8.9991e+12
9.999e+12
示例 2:
#include <iostream>
class A
{
public:
A() {
for(int i = 0; i < 10000; ++i)
for(int j = 0; j < 10000; ++j)
m += i+j;
}
~A() {}
double m;
};
void foo()
{
const static A a;
std::cout << a.m << std::endl;
}//foo
int main()
{
for(int i = 0; i < 10; ++i)
foo();
return 0;
}//main
结果:
9.999e+11
9.999e+11
9.999e+11
9.999e+11
9.999e+11
9.999e+11
9.999e+11
9.999e+11
9.999e+11
9.999e+11
示例
1 和示例 2 之间的唯一区别是"静态"。
我承认第一个常量并没有像我预期的那样起作用。
提前谢谢。
伙计们,你们说得太对了!我该死。所以。。。
更重要的是,使用const static
本地对象总是比只使用const
本地对象更好吗?
您有未定义的行为,因为数据成员A::m
在写入此处之前尚未初始化:
m += i+j;
在写入它之前,您应该将其初始化为有效值。 0
似乎是一个不错的选择:
A() : m(0) {
for(int i = 0; i < 10000; ++i)
for(int j = 0; j < 10000; ++j)
m += i+j;
}
声明 static A
实例时,所有数据成员都将初始化为零,并且在程序流首次到达声明时调用构造函数。这会屏蔽未初始化的数据成员的效果。
请注意,在 C++11 中,您还可以在声明点进行初始化:
double m = 0;
所有静态对象都是零初始化的。因此,在第二种情况下,数据成员 m 的初始值为 0。
在第一种情况下,类 A 的对象是默认初始化的。由于您没有在构造函数中初始化 m,因此它有一个未指定的(任意)值。
这解释了差异。
你应该初始化成员:
A():m(0) {
// stuff
}
相关文章:
- 为不同配置设置MSVC_RUNTIME_LIBRARY的正确方法是什么
- 警告处理为错误这里有什么问题
- 什么时候调用组成单元对象的析构函数
- #定义c-预处理器常量..我做错了什么
- 努力将整数转换为链表。不知道我在这里做错了什么
- C++我的数学有什么问题,为什么我的代码不能正确循环
- 什么时候在C++中返回常量引用是个好主意
- 当在同一名称空间中有两个具有相同签名的函数时,会发生什么
- C++避免重复声明的语法是什么
- c++库的公共头文件中应该包含什么
- 问题:什么是QAbstractItemView::NoEditTriggers的反面
- 有什么方法可以遍历结构吗
- 当类在C++中定义时,有什么方法可以"register"类吗?
- ifstream什么都没读
- 在C++中,将大的无符号浮点数四舍五入为整数的最佳方法是什么
- 实现无开销push_back的最佳方法是什么
- C++从另一个类访问公共静态向量的正确方法是什么
- "throw expression code" 1e7 >返回 d 是什么?投掷标准::overflow_error( "too big" ) : d;意味 着?
- 我应该使用什么来代替void作为变体中的替代类型之一
- 有没有什么方法可以使用一个函数中定义的常量变量,也可以由c++中同一程序中的其他函数使用