Skip to content

Instantly share code, notes, and snippets.

@mdmaas
Created April 5, 2023 06:44
Show Gist options
  • Star 1 You must be signed in to star a gist
  • Fork 1 You must be signed in to fork a gist
  • Save mdmaas/e5fb73abf8a17afe2ddf09d1ebbecb11 to your computer and use it in GitHub Desktop.
Save mdmaas/e5fb73abf8a17afe2ddf09d1ebbecb11 to your computer and use it in GitHub Desktop.
Simple performance benchmark of stack, heap, and pre-allocated arrays and std::vector
// g++ vector_vs_stack_array.cpp -O3 -o vector
#include <numeric>
#include <vector>
#include <chrono>
#include <iostream>
#include <cmath>
using namespace std::chrono;
double sumNewArray( int N ) {
double * smallarray = new double[N];
for(unsigned int k = 0; k<N; k++){
smallarray[k] = 1.0/pow(k+1,2);
}
double sum = 0.0;
for(unsigned int k = 0; k<N; k++){
sum += smallarray[k];
}
delete [] smallarray;
return sum;
}
double sumStackArray( int N ) {
double smallarray[N];
for(unsigned int k = 0; k<N; k++){
smallarray[k] = 1.0/pow(k+1,2);
}
double sum = 0.0;
for(unsigned int k = 0; k<N; k++){
sum += smallarray[k];
}
return sum;
}
double sumStdVector( int N ) {
std::vector<double> smallarray;
smallarray.reserve(N);
for(unsigned int k = 0; k<N; ++k){
smallarray[k] = 1.0/pow(k+1,2);
}
double sum = 0.0;
for(unsigned int k = 0; k<N; k++){
sum += smallarray[k];
}
return sum;
}
double sumVector_PushBack( int N ) {
std::vector<double> smallarray;
for(unsigned int k = 0; k<N; ++k){
smallarray.push_back(1.0/pow(k+1,2));
}
double sum = 0.0;
for(unsigned int k = 0; k<N; k++){
sum += smallarray[k];
}
return sum;
}
//
// With pre-allocation
//
double sumArray_prealloc( int N, double smallarray[] ) {
for(unsigned int k = 0; k<N; ++k){
smallarray[k] = 1.0/pow(k+1,2);
}
double sum = 0.0;
for(unsigned int k = 0; k<N; k++){
sum += smallarray[k];
}
return sum;
}
double sumStdVector_prealloc( std::vector<double>& smallarray ) {
int N = smallarray.size();
for(unsigned int k = 0; k<N; ++k){
smallarray[k] = 1.0/pow(k+1,2);
}
double sum = 0.0;
for(unsigned int k = 0; k<N; k++){
sum += smallarray[k];
}
return sum;
}
//
// Timing functions
//
int time_sumNewArray( int N, int REP ) {
double x = 0.0;
auto start = high_resolution_clock::now();
for (int i=0; i<REP; i++){
x = sumNewArray(N);
}
auto stop = high_resolution_clock::now();
int t = duration_cast<microseconds>(stop - start).count();
return t;
}
int time_sumStackArray( int N, int REP ) {
double x = 0.0;
auto start = high_resolution_clock::now();
for (int i=0; i<REP; i++){
x = sumStackArray(N);
}
auto stop = high_resolution_clock::now();
int t = duration_cast<microseconds>(stop - start).count();
return t;
}
int time_sumStdVector( int N, int REP ) {
double x = 0.0;
auto start = high_resolution_clock::now();
for (int i=0; i<REP; i++){
x = sumStdVector(N);
}
auto stop = high_resolution_clock::now();
int t = duration_cast<microseconds>(stop - start).count();
return t;
}
int time_sumStdVector_prealloc( int N, int REP ) {
double x = 0.0;
auto start = high_resolution_clock::now();
std::vector<double> smallarray(N);
// smallarray.reserve(N);
for (int i=0; i<REP; i++){
x = sumStdVector_prealloc(smallarray);
}
auto stop = high_resolution_clock::now();
int t = duration_cast<microseconds>(stop - start).count();
return t;
}
int time_sumNewArray_prealloc( int N, int REP ) {
double x = 0.0;
auto start = high_resolution_clock::now();
double * smallarray = new double[N];
// smallarray.reserve(N);
for (int i=0; i<REP; i++){
x = sumArray_prealloc( N, smallarray );
}
auto stop = high_resolution_clock::now();
int t = duration_cast<microseconds>(stop - start).count();
return t;
}
int time_sumStackArray_prealloc( int N, int REP ) {
double x = 0.0;
auto start = high_resolution_clock::now();
double smallarray[N];
// smallarray.reserve(N);
for (int i=0; i<REP; i++){
x = sumArray_prealloc( N, smallarray );
}
auto stop = high_resolution_clock::now();
int t = duration_cast<microseconds>(stop - start).count();
return t;
}
int time_sumVector_pushback( int N, int REP ) {
double x = 0.0;
auto start = high_resolution_clock::now();
for (int i=0; i<REP; i++){
x = sumVector_PushBack(N);
}
auto stop = high_resolution_clock::now();
int t = duration_cast<microseconds>(stop - start).count();
return t;
}
int printVec( std::string varname, std::vector<int> values )
{
std::cout << varname;
for (int i=0; i< values.size(); i++) {
std::cout << " | " << values[i];
}
std::cout << std::endl;
return 0;
}
int main(){
int REP = 1000;
std::vector<int> Ns = {10, 100, 1000, 10000};
std::vector<int> t(Ns.size());
freopen("output.txt","w",stdout);
std::cout << "Function | Miliseconds" << std::endl;
// Warm up memory?
t[0] = time_sumNewArray(20000, REP);
for (int i=0; i<Ns.size(); i++) {
t[i] = time_sumStackArray(Ns[i], REP);
}
printVec("StackArray", t);
for (int i=0; i<Ns.size(); i++) {
t[i] = time_sumNewArray(Ns[i], REP);
}
printVec("NewArray", t);
for (int i=0; i<Ns.size(); i++) {
t[i] = time_sumStdVector(Ns[i], REP);
}
printVec("StdVector", t);
for (int i=0; i<Ns.size(); i++) {
t[i] = time_sumVector_pushback(Ns[i], REP);
}
printVec("StdVector_pushback", t);
for (int i=0; i<Ns.size(); i++) {
t[i] = time_sumVector_Eigen_VectorXd(Ns[i], REP);
}
printVec("Eigen_VectorXd", t);
for (int i=0; i<Ns.size(); i++) {
t[i] = time_sumStackArray_prealloc(Ns[i], REP);
}
printVec("StackArray_prealloc", t);
for (int i=0; i<Ns.size(); i++) {
t[i] = time_sumNewArray_prealloc(Ns[i], REP);
}
printVec("NewArray_prealloc", t);
for (int i=0; i<Ns.size(); i++) {
t[i] = time_sumStdVector_prealloc(Ns[i], REP);
}
printVec("StdVector_prealloc", t);
return 0;
}
Sign up for free to join this conversation on GitHub. Already have an account? Sign in to comment