2016-12-23 02:28:21 +08:00
|
|
|
#include "gtest/gtest.h"
|
2016-08-09 18:53:10 +08:00
|
|
|
|
2017-03-21 20:48:35 +08:00
|
|
|
#include <memory>
|
2016-08-09 18:53:10 +08:00
|
|
|
#include "data_structures/map/rh_hashmap.hpp"
|
|
|
|
|
2017-02-18 18:54:37 +08:00
|
|
|
class Data {
|
|
|
|
private:
|
|
|
|
int key;
|
2016-08-09 18:53:10 +08:00
|
|
|
|
2017-02-18 18:54:37 +08:00
|
|
|
public:
|
|
|
|
Data(int key) : key(key) {}
|
2016-08-09 18:53:10 +08:00
|
|
|
|
2017-02-18 18:54:37 +08:00
|
|
|
const int &get_key() const { return key; }
|
2016-08-09 18:53:10 +08:00
|
|
|
};
|
|
|
|
|
2016-08-11 03:02:54 +08:00
|
|
|
void cross_validate(RhHashMap<int, Data> &map, std::map<int, Data *> &s_map);
|
|
|
|
|
2017-02-18 18:54:37 +08:00
|
|
|
TEST(RobinHoodHashmap, BasicFunctionality) {
|
|
|
|
RhHashMap<int, Data> map;
|
2016-08-09 18:53:10 +08:00
|
|
|
|
2017-02-18 18:54:37 +08:00
|
|
|
ASSERT_EQ(map.size(), 0);
|
2017-03-21 20:48:35 +08:00
|
|
|
Data d0(0);
|
|
|
|
ASSERT_EQ(map.insert(&d0), true);
|
2017-02-18 18:54:37 +08:00
|
|
|
ASSERT_EQ(map.size(), 1);
|
2016-08-09 18:53:10 +08:00
|
|
|
}
|
|
|
|
|
2017-02-18 18:54:37 +08:00
|
|
|
TEST(RobinHoodHashmap, RemoveFunctionality) {
|
|
|
|
RhHashMap<int, Data> map;
|
2016-08-11 03:02:54 +08:00
|
|
|
|
2017-03-21 20:48:35 +08:00
|
|
|
Data d0(0);
|
|
|
|
ASSERT_EQ(map.insert(&d0), true);
|
2017-02-18 18:54:37 +08:00
|
|
|
ASSERT_EQ(map.remove(0).is_present(), true);
|
|
|
|
ASSERT_EQ(map.size(), 0);
|
|
|
|
ASSERT_EQ(!map.find(0).is_present(), true);
|
2016-08-11 03:02:54 +08:00
|
|
|
}
|
|
|
|
|
2017-02-18 18:54:37 +08:00
|
|
|
TEST(RobinHoodHashmap, InsertGetCheck) {
|
|
|
|
RhHashMap<int, Data> map;
|
2016-08-09 18:53:10 +08:00
|
|
|
|
2017-02-18 18:54:37 +08:00
|
|
|
ASSERT_EQ(!map.find(0).is_present(), true);
|
2017-03-21 20:48:35 +08:00
|
|
|
Data d0(0);
|
|
|
|
ASSERT_EQ(map.insert(&d0), true);
|
2017-02-18 18:54:37 +08:00
|
|
|
ASSERT_EQ(map.find(0).is_present(), true);
|
2017-03-21 20:48:35 +08:00
|
|
|
ASSERT_EQ(map.find(0).get(), &d0);
|
2016-08-09 18:53:10 +08:00
|
|
|
}
|
|
|
|
|
2017-02-18 18:54:37 +08:00
|
|
|
TEST(RobinHoodHashmap, DoubleInsert) {
|
|
|
|
RhHashMap<int, Data> map;
|
2016-08-09 18:53:10 +08:00
|
|
|
|
2017-03-21 20:48:35 +08:00
|
|
|
Data d0(0);
|
|
|
|
ASSERT_EQ(map.insert(&d0), true);
|
|
|
|
ASSERT_EQ(!map.insert(&d0), true);
|
2016-08-09 18:53:10 +08:00
|
|
|
}
|
|
|
|
|
2017-02-18 18:54:37 +08:00
|
|
|
TEST(RobinHoodHashmap, FindInsertFind) {
|
|
|
|
RhHashMap<int, Data> map;
|
2016-08-09 18:53:10 +08:00
|
|
|
|
2017-03-21 20:48:35 +08:00
|
|
|
std::vector<std::unique_ptr<Data>> di;
|
|
|
|
di.reserve(128);
|
|
|
|
for (int i = 0; i < 128; ++i) di.emplace_back(std::make_unique<Data>(i));
|
2017-02-18 18:54:37 +08:00
|
|
|
for (int i = 0; i < 128; i++) {
|
|
|
|
ASSERT_EQ(!map.find(i).is_present(), true);
|
2017-03-21 20:48:35 +08:00
|
|
|
ASSERT_EQ(map.insert(di[i].get()), true);
|
2017-02-18 18:54:37 +08:00
|
|
|
ASSERT_EQ(map.find(i).is_present(), true);
|
|
|
|
}
|
2016-08-09 18:53:10 +08:00
|
|
|
|
2017-02-18 18:54:37 +08:00
|
|
|
for (int i = 0; i < 128; i++) {
|
|
|
|
ASSERT_EQ(map.find(i).is_present(), true);
|
|
|
|
ASSERT_EQ(map.find(i).get()->get_key(), i);
|
|
|
|
}
|
2016-08-09 23:44:39 +08:00
|
|
|
}
|
|
|
|
|
2017-02-18 18:54:37 +08:00
|
|
|
TEST(RobinHoodHashmap, Iterate) {
|
|
|
|
RhHashMap<int, Data> map;
|
|
|
|
|
2017-03-21 20:48:35 +08:00
|
|
|
std::vector<std::unique_ptr<Data>> di;
|
|
|
|
di.reserve(128);
|
|
|
|
for (int i = 0; i < 128; ++i) di.emplace_back(std::make_unique<Data>(i));
|
2017-02-18 18:54:37 +08:00
|
|
|
for (int i = 0; i < 128; i++) {
|
|
|
|
ASSERT_EQ(!map.find(i).is_present(), true);
|
2017-03-21 20:48:35 +08:00
|
|
|
ASSERT_EQ(map.insert(di[i].get()), true);
|
2017-02-18 18:54:37 +08:00
|
|
|
ASSERT_EQ(map.find(i).is_present(), true);
|
|
|
|
}
|
|
|
|
|
|
|
|
bool seen[128] = {false};
|
|
|
|
for (auto e : map) {
|
|
|
|
auto key = e->get_key();
|
|
|
|
ASSERT_EQ(!seen[key], true);
|
|
|
|
seen[key] = true;
|
|
|
|
}
|
|
|
|
for (int i = 0; i < 128; i++) {
|
|
|
|
ASSERT_EQ(seen[i], true);
|
|
|
|
}
|
2016-08-09 23:44:39 +08:00
|
|
|
}
|
|
|
|
|
2017-02-18 18:54:37 +08:00
|
|
|
TEST(RobinHoodHashmap, Checked) {
|
|
|
|
RhHashMap<int, Data> map;
|
|
|
|
std::map<int, Data *> s_map;
|
|
|
|
|
2017-03-21 20:48:35 +08:00
|
|
|
std::vector<std::unique_ptr<Data>> di;
|
|
|
|
std::vector<int> key;
|
|
|
|
di.reserve(128);
|
|
|
|
key.reserve(128);
|
|
|
|
for (int i = 0; i < 128; ++i) {
|
|
|
|
const int curr_key = std::rand();
|
|
|
|
key.emplace_back(curr_key);
|
|
|
|
di.emplace_back(std::make_unique<Data>(curr_key));
|
|
|
|
}
|
2017-02-18 18:54:37 +08:00
|
|
|
for (int i = 0; i < 128; i++) {
|
2017-03-21 20:48:35 +08:00
|
|
|
if (map.insert(di[i].get())) {
|
|
|
|
ASSERT_EQ(s_map.find(key[i]), s_map.end());
|
|
|
|
s_map[key[i]] = di[i].get();
|
2017-02-18 18:54:37 +08:00
|
|
|
} else {
|
2017-03-21 20:48:35 +08:00
|
|
|
ASSERT_NE(s_map.find(key[i]), s_map.end());
|
2016-08-09 23:44:39 +08:00
|
|
|
}
|
2017-02-18 18:54:37 +08:00
|
|
|
}
|
2016-08-09 23:44:39 +08:00
|
|
|
|
2017-02-18 18:54:37 +08:00
|
|
|
cross_validate(map, s_map);
|
2016-08-11 03:02:54 +08:00
|
|
|
}
|
|
|
|
|
2017-02-18 18:54:37 +08:00
|
|
|
TEST(RobinHoodHashMap, CheckWithRemove) {
|
|
|
|
RhHashMap<int, Data> map;
|
|
|
|
std::map<int, Data *> s_map;
|
2017-03-21 20:48:35 +08:00
|
|
|
std::vector<std::unique_ptr<Data>> di;
|
|
|
|
std::vector<int> key;
|
|
|
|
di.reserve(1280);
|
|
|
|
key.reserve(1280);
|
|
|
|
for (int i = 0; i < 1280; ++i) {
|
|
|
|
const int curr_key = std::rand() % 100;
|
|
|
|
key.emplace_back(curr_key);
|
|
|
|
di.emplace_back(std::make_unique<Data>(curr_key));
|
|
|
|
}
|
2017-02-18 18:54:37 +08:00
|
|
|
|
|
|
|
for (int i = 0; i < 1280; i++) {
|
2017-03-21 20:48:35 +08:00
|
|
|
if (map.insert(di[i].get())) {
|
|
|
|
ASSERT_EQ(s_map.find(key[i]), s_map.end());
|
|
|
|
s_map[key[i]] = di[i].get();
|
2017-02-18 18:54:37 +08:00
|
|
|
cross_validate(map, s_map);
|
|
|
|
} else {
|
2017-03-21 20:48:35 +08:00
|
|
|
ASSERT_EQ(map.remove(key[i]).is_present(), true);
|
|
|
|
ASSERT_EQ(s_map.erase(key[i]), 1);
|
2017-02-18 18:54:37 +08:00
|
|
|
cross_validate(map, s_map);
|
2016-08-11 03:02:54 +08:00
|
|
|
}
|
2017-02-18 18:54:37 +08:00
|
|
|
}
|
2016-08-11 03:02:54 +08:00
|
|
|
|
2017-02-18 18:54:37 +08:00
|
|
|
cross_validate(map, s_map);
|
2016-08-11 03:02:54 +08:00
|
|
|
}
|
|
|
|
|
2017-03-21 20:48:35 +08:00
|
|
|
TEST(RobinhoodHashmmap, AlignmentCheck) {
|
|
|
|
RhHashMap<int, Data> map;
|
|
|
|
char *block = static_cast<char *>(std::malloc(20));
|
|
|
|
++block; // not alligned - offset 1
|
|
|
|
EXPECT_DEATH(map.insert((Data *)(block)), "not 8-alligned");
|
|
|
|
}
|
|
|
|
|
2017-02-18 18:54:37 +08:00
|
|
|
void cross_validate(RhHashMap<int, Data> &map, std::map<int, Data *> &s_map) {
|
|
|
|
for (auto e : map) {
|
|
|
|
ASSERT_NE(s_map.find(e->get_key()), s_map.end());
|
|
|
|
}
|
2016-08-09 23:44:39 +08:00
|
|
|
|
2017-02-18 18:54:37 +08:00
|
|
|
for (auto e : s_map) {
|
|
|
|
ASSERT_EQ(map.find(e.first).get(), e.second);
|
|
|
|
}
|
2016-08-09 18:53:10 +08:00
|
|
|
}
|
2016-12-23 02:28:21 +08:00
|
|
|
|
2017-02-18 18:54:37 +08:00
|
|
|
int main(int argc, char **argv) {
|
|
|
|
::testing::InitGoogleTest(&argc, argv);
|
|
|
|
return RUN_ALL_TESTS();
|
2016-12-23 02:28:21 +08:00
|
|
|
}
|