Недавно я столкнулся с шаблонами битов и хотел бы использовать их в своем текущем проекте. Я std::bitset шаблон std::bitset должен иметь размер, определенный во время компиляции. Многие предлагают использовать boost::dynamic_bitset чтобы облегчить это требование.
Чтобы сравнить эти два, я решил сделать сравнение скорости методов set, flip и count.
Результаты довольно странные... и мне интересно, может ли кто-нибудь пролить свет на него для меня.
Код находится в конце сообщения, но я объясню, что я здесь делаю. У меня есть один объект std::bitset (назовите его bs) и один объект boost::dynamic_bitset (назовите его dynbs). Каждый имеет n=1000000 бит. Для данного метода выше, вызовите метод на каждом из n бит последовательно и повторите это R=10000 раз.
Используя библиотеку std::chrono, здесь приведены тайминги для каждого в наносекундах:
set
bitset: 267 nsecs
dyn bitset: 18603174546 nsecs
flip
bitset: 73 nsecs
dyn bitset: 18842352867 nsecs
count
bitset: 77 nsecs
dyn bitset: 51 nsecs
Функция boost::dynamic_bitset кажется намного медленнее для set и flip.
Для того, чтобы сделать его более интересным, если метод reset вызываются два объектов до выполнения этих тестов, то тайминги сопоставимы. Вот они:
set
bitset: 19397779399 nsecs
dyn bitset: 18472863864 nsecs
flip
bitset: 18599248629 nsecs
dyn bitset: 18376267939 nsecs
count
bitset: 68 nsecs
dyn bitset: 61 nsecs
Теперь оба контейнера заявляют, что инициализируют все биты до 0, поэтому вызов reset не должен изменять какой-либо бит. Демпинг выхода none до и после reset действительно подтверждает это.
Поэтому после всего этого у меня есть два вопроса:
1) Почему boost::dynamic_bitset намного медленнее, чем std::bitset при вызове set и flip?
2) Почему reset вызова оказывает огромное негативное влияние на скорость std::bitset?
Вот мой код:
#include <iostream>
#include <iomanip>
#include <bitset>
#include <boost/dynamic_bitset.hpp>
#include <vector>
#include <chrono>
#include <ctime>
using namespace std;
using namespace chrono;
using namespace boost;
int main(){
const unsigned int n=1000000;
bitset< n > bs;
dynamic_bitset< > dynbs(n);
// bs.reset();
// dynbs.reset();
unsigned int i,r,R=10000;
high_resolution_clock::time_point tick,tock;
////////////////////////////////////////////////////////////
// Method: set
std::cout << "set" << std::endl;
tick=high_resolution_clock::now();
for(r=0; r<R; r++)
for(i=0; i<n; i++)
bs.set(i);
tock=high_resolution_clock::now();
cout << setw(16) << "bitset: "
<< setw(16) << duration_cast<nanoseconds>(tock-tick).count() << " nsecs"
<< std::endl;
tick=high_resolution_clock::now();
for(r=0; r<R; r++)
for(i=0; i<n; i++)
dynbs.set(i);
tock=high_resolution_clock::now();
cout << setw(16) << "dyn bitset: "
<< setw(16) << duration_cast<nanoseconds>(tock-tick).count() << " nsecs"
<< std::endl << std::endl;
////////////////////////////////////////////////////////////
// Method: flip
std::cout << "flip" << std::endl;
tick=high_resolution_clock::now();
for(r=0; r<R; r++)
for(i=0; i<n; i++)
bs.flip(i);
tock=high_resolution_clock::now();
cout << setw(16) << "bitset: "
<< setw(16) << duration_cast<nanoseconds>(tock-tick).count() << " nsecs"
<< std::endl;
tick=high_resolution_clock::now();
for(r=0; r<R; r++)
for(i=0; i<n; i++)
dynbs.flip(i);
tock=high_resolution_clock::now();
cout << setw(16) << "dyn bitset: "
<< setw(16) << duration_cast<nanoseconds>(tock-tick).count() << " nsecs"
<< std::endl << std::endl;
////////////////////////////////////////////////////////////
// Method: count
std::cout << "count" << std::endl;
tick=high_resolution_clock::now();
for(r=0; r<R; r++)
for(i=0; i<n; i++)
bs.count();
tock=high_resolution_clock::now();
cout << setw(16) << "bitset: "
<< setw(16) << duration_cast<nanoseconds>(tock-tick).count() << " nsecs"
<< std::endl;
tick=high_resolution_clock::now();
for(r=0; r<R; r++)
for(i=0; i<n; i++)
dynbs.count();
tock=high_resolution_clock::now();
cout << setw(16) << "dyn bitset: "
<< setw(16) << duration_cast<nanoseconds>(tock-tick).count() << " nsecs"
<< std::endl;
return 0;
}
Я скомпилировал его с помощью
g++ -O3 -std=c++11 bitset.cpp -o bitset
где bitset.cpp - это код, вставленный выше.