- html - 出于某种原因,IE8 对我的 Sass 文件中继承的 html5 CSS 不友好?
- JMeter 在响应断言中使用 span 标签的问题
- html - 在 :hover and :active? 上具有不同效果的 CSS 动画
- html - 相对于居中的 html 内容固定的 CSS 重复背景?
我注意到有时程序运行很慢,但后来性能很好。例如,我有一些循环运行的代码,第一次迭代需要很长时间,但相同代码的其他迭代运行得非常快。很难说出具体情况,因为我无法弄清楚,而且似乎即使是单个文字也会影响这种行为。我准备了一个小代码片段:
#include <chrono>
#include <vector>
#include <iostream>
using namespace std;
int main()
{
const int num{ 100000 };
vector<vector<int>> octs;
for (int i{ 0 }; i < num; ++i)
{
octs.emplace_back(vector<int>{ 42 });
}
vector<int> datas;
for (int i{ 0 }; i < num; ++i)
{
datas.push_back(42);
}
for (int n{ 0 }; n < 10; ++n)
{
cout << "start" << '\n';
//cout << 0 << "start" << '\n';
auto start = chrono::high_resolution_clock::now();
for (int i{ 0 }; i < num; ++i)
{
vector<int> points{ 42 };
}
auto end = chrono::high_resolution_clock::now();
auto time = chrono::duration_cast<chrono::milliseconds>(end - start);
cout << time.count() << '\n';
}
cin.get();
return 0;
}
前两个 vector 是必不可少的。至少在 Visual Studio 中。以为它们没有被使用,它们会对性能产生很大影响。此外,调整它们也会产生性能影响(例如更改初始化顺序,删除 push_back
并在构造函数中分配必要的大小)。但是这段代码给了我以下结果:
此外,对于 vs2013,如果我取消注释行 cout << 0 << "start" << '\n';
性能问题消失了,所有迭代都是平等的!
这是怎么回事?
最佳答案
对于前两个循环,最大的性能考虑可能是内存分配,以及将 vector
内容复制到更大的缓冲区。在这种情况下,循环似乎“加速”这一事实并不奇怪。
这是由于 vector
类的实现细节所致。让我们看看documentation :
Internally, vectors use a dynamically allocated array to store their elements. This array may need to be reallocated in order to grow in size when new elements are inserted, which implies allocating a new array and moving all elements to it. This is a relatively expensive task in terms of processing time, and thus, vectors do not reallocate each time an element is added to the container.
Instead, vector containers may allocate some extra storage to accommodate for possible growth, and thus the container may have an actual capacity greater than the storage strictly needed to contain its elements (i.e., its size). Libraries can implement different strategies for growth to balance between memory usage and reallocations, but in any case, reallocations should only happen at logarithmically growing intervals of size so that the insertion of individual elements at the end of the vector can be provided with amortized constant time complexity (see push_back).
所以在幕后,为您的 vector
分配的实际内存可能比您实际使用的要多得多。因此,当您向 vector
添加一个不适合其当前缓冲区的新元素时,vector
只需要进行代价高昂的重新分配和复制。此外,由于它表示重新分配应该只以对数增长的间隔发生,您可以预期 vector
类在每次需要重新分配时都会将缓冲区大小大致加倍。但请注意,各种平台上的 vector
实现都经过高度调整,以针对该类的最常见使用模式进行优化,这可能是您在工具链和平台上看到的不同性能的一个因素。
所以你应该看到循环在前几次执行时很慢,然后随着 push_back
和 emplace
操作需要做更少的重新分配和复制而获得更快的速度以适应新元素。
所以我认为这是您可以用来推断执行前两个循环需要多长时间的主要事实。但是对于您的具体示例,由于程序的简单性,编译器可能会对其生成的代码采取一些自由。所以我们可以想象,一个足够聪明的优化编译器可能能够看到您的 vector
只会增长到它在编译时知道的大小,num
。这是我怀疑你最后一个循环的最大问题,这似乎是一个任意且无用的测试。例如,循环 3 中的嵌套循环可以完全优化掉。我认为这是您在不同编译器中看到如此不同的运行时行为的主要原因。
如果您想了解真实情况,请查看您的编译器生成的汇编代码。
关于c++ - 为什么程序启动缓慢但后来获得全速?,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/38482007/
我有一个存储库,其中包含我通过在本地主机上运行服务器进行测试的代码。标准程序是启动 Apache-Tomcat-7,使用 Maven 重新构建项目并进行部署。 经过一段时间的编码,我决定 git st
我是一名优秀的程序员,十分优秀!