2020-07-24 05:25:44 +08:00
|
|
|
/**
|
|
|
|
* @file chaining.cpp
|
|
|
|
* @author [vasutomar](https://github.com/vasutomar)
|
|
|
|
* @author [Krishna Vedala](https://github.com/kvedala)
|
2020-07-31 06:30:02 +08:00
|
|
|
* @brief Implementation of [hash
|
|
|
|
* chains](https://en.wikipedia.org/wiki/Hash_chain).
|
2020-07-24 05:25:44 +08:00
|
|
|
*/
|
|
|
|
#include <cmath>
|
2020-06-20 00:04:56 +08:00
|
|
|
#include <iostream>
|
2020-07-24 05:25:44 +08:00
|
|
|
#include <memory>
|
|
|
|
#include <vector>
|
2020-06-20 00:04:56 +08:00
|
|
|
|
2020-07-24 05:25:44 +08:00
|
|
|
/**
|
|
|
|
* @brief Chain class with a given modulus
|
|
|
|
*/
|
2020-07-31 06:30:02 +08:00
|
|
|
class hash_chain {
|
2020-07-24 05:25:44 +08:00
|
|
|
private:
|
2020-07-31 06:30:02 +08:00
|
|
|
/**
|
|
|
|
* @brief Define a linked node
|
|
|
|
*/
|
|
|
|
using Node = struct Node {
|
|
|
|
int data{}; ///< data stored in the node
|
|
|
|
std::shared_ptr<struct Node> next; ///< pointer to the next node
|
|
|
|
};
|
|
|
|
|
2020-07-31 06:39:24 +08:00
|
|
|
std::vector<std::shared_ptr<Node>> head; ///< array of nodes
|
2021-08-18 01:25:08 +08:00
|
|
|
int _mod; ///< modulus of the class
|
2020-06-20 00:04:56 +08:00
|
|
|
|
2020-07-24 05:25:44 +08:00
|
|
|
public:
|
|
|
|
/**
|
|
|
|
* @brief Construct a new chain object
|
|
|
|
*
|
|
|
|
* @param mod modulus of the chain
|
|
|
|
*/
|
2020-07-31 06:30:02 +08:00
|
|
|
explicit hash_chain(int mod) : _mod(mod) {
|
2020-07-24 05:25:44 +08:00
|
|
|
while (mod--) head.push_back(nullptr);
|
2020-06-20 00:04:56 +08:00
|
|
|
}
|
|
|
|
|
2020-07-24 05:25:44 +08:00
|
|
|
/**
|
|
|
|
* @brief create and add a new node with a give value and at a given height
|
|
|
|
*
|
|
|
|
* @param x value at the new node
|
|
|
|
* @param h height of the node
|
|
|
|
*/
|
|
|
|
void add(int x, int h) {
|
2020-07-31 06:39:24 +08:00
|
|
|
std::shared_ptr<Node> curr;
|
|
|
|
std::shared_ptr<Node> temp(new Node);
|
2020-07-24 05:25:44 +08:00
|
|
|
temp->data = x;
|
|
|
|
temp->next = nullptr;
|
|
|
|
if (!head[h]) {
|
|
|
|
head[h] = temp;
|
|
|
|
curr = head[h];
|
2020-06-20 00:04:56 +08:00
|
|
|
} else {
|
2020-07-24 05:25:44 +08:00
|
|
|
curr = head[h];
|
|
|
|
while (curr->next) curr = curr->next;
|
|
|
|
curr->next = temp;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* @brief Display the chain
|
|
|
|
*/
|
|
|
|
void display() {
|
2020-07-31 06:39:24 +08:00
|
|
|
std::shared_ptr<Node> temp = nullptr;
|
2020-07-24 05:25:44 +08:00
|
|
|
int i = 0;
|
|
|
|
for (i = 0; i < _mod; i++) {
|
|
|
|
if (!head[i]) {
|
|
|
|
std::cout << "Key " << i << " is empty" << std::endl;
|
|
|
|
} else {
|
2021-08-16 00:23:18 +08:00
|
|
|
std::cout << "Key " << i << " has values = " << std::endl;
|
2020-07-24 05:25:44 +08:00
|
|
|
temp = head[i];
|
|
|
|
while (temp->next) {
|
2021-08-16 00:23:18 +08:00
|
|
|
std::cout << temp->data << " " << std::endl;
|
2020-07-24 05:25:44 +08:00
|
|
|
temp = temp->next;
|
|
|
|
}
|
|
|
|
std::cout << temp->data;
|
|
|
|
std::cout << std::endl;
|
2020-06-20 00:04:56 +08:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2020-07-24 05:25:44 +08:00
|
|
|
/**
|
|
|
|
* @brief Compute the hash of a value for current chain
|
|
|
|
*
|
|
|
|
* @param x value to compute modulus of
|
|
|
|
* @return modulus of `x`
|
2020-07-31 06:30:02 +08:00
|
|
|
* @note declared as a
|
|
|
|
* [`virtual`](https://en.cppreference.com/w/cpp/language/virtual) so that
|
|
|
|
* custom implementations of the class can modify the hash function.
|
2020-07-24 05:25:44 +08:00
|
|
|
*/
|
2020-07-31 06:30:02 +08:00
|
|
|
virtual int hash(int x) const { return x % _mod; }
|
2020-06-20 00:04:56 +08:00
|
|
|
|
2020-07-31 06:30:02 +08:00
|
|
|
/**
|
|
|
|
* @brief Find if a value and corresponding hash exist
|
|
|
|
*
|
|
|
|
* @param x value to search for
|
|
|
|
* @param h corresponding hash key
|
|
|
|
* @returns `true` if element found
|
|
|
|
* @returns `false` if element not found
|
|
|
|
*/
|
|
|
|
bool find(int x, int h) const {
|
2020-07-31 06:39:24 +08:00
|
|
|
std::shared_ptr<Node> temp = head[h];
|
2020-07-24 05:25:44 +08:00
|
|
|
if (!head[h]) {
|
2020-07-31 06:30:02 +08:00
|
|
|
// index does not exist!
|
2021-08-16 00:23:18 +08:00
|
|
|
std::cout << "Element not found" << std::endl;
|
2020-07-31 06:30:02 +08:00
|
|
|
return false;
|
2020-07-24 05:25:44 +08:00
|
|
|
}
|
2020-07-31 06:30:02 +08:00
|
|
|
|
|
|
|
// scan for data value
|
2020-07-24 05:25:44 +08:00
|
|
|
while (temp->data != x && temp->next) temp = temp->next;
|
2020-07-31 06:30:02 +08:00
|
|
|
|
2020-07-24 05:25:44 +08:00
|
|
|
if (temp->next) {
|
2021-08-16 00:23:18 +08:00
|
|
|
std::cout << "Element found" << std::endl;
|
2020-07-31 06:30:02 +08:00
|
|
|
return true;
|
2020-07-24 05:25:44 +08:00
|
|
|
}
|
2020-07-31 06:30:02 +08:00
|
|
|
|
|
|
|
// implicit else condition
|
|
|
|
// i.e., temp->next == nullptr
|
|
|
|
if (temp->data == x) {
|
2021-08-16 00:23:18 +08:00
|
|
|
std::cout << "Element found" << std::endl;
|
2020-07-31 06:30:02 +08:00
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
|
|
|
// further implicit else condition
|
2021-08-16 00:23:18 +08:00
|
|
|
std::cout << "Element not found" << std::endl;
|
2020-07-31 06:30:02 +08:00
|
|
|
return false;
|
2020-06-20 00:04:56 +08:00
|
|
|
}
|
2020-07-24 05:25:44 +08:00
|
|
|
};
|
|
|
|
|
|
|
|
/** Main function
|
|
|
|
* @returns `0` always
|
|
|
|
*/
|
|
|
|
int main() {
|
|
|
|
int c = 0, x = 0, mod = 0, h = 0;
|
2021-08-16 00:23:18 +08:00
|
|
|
std::cout << "Enter the size of Hash Table. = " << std::endl;
|
2020-07-24 05:25:44 +08:00
|
|
|
std::cin >> mod;
|
|
|
|
|
2020-07-31 06:30:02 +08:00
|
|
|
hash_chain mychain(mod);
|
2020-06-20 00:04:56 +08:00
|
|
|
|
|
|
|
bool loop = true;
|
|
|
|
while (loop) {
|
2020-07-24 05:25:44 +08:00
|
|
|
std::cout << std::endl;
|
|
|
|
std::cout << "PLEASE CHOOSE -" << std::endl;
|
|
|
|
std::cout << "1. Add element." << std::endl;
|
|
|
|
std::cout << "2. Find element." << std::endl;
|
|
|
|
std::cout << "3. Generate Hash." << std::endl;
|
|
|
|
std::cout << "4. Display Hash table." << std::endl;
|
|
|
|
std::cout << "5. Exit." << std::endl;
|
|
|
|
std::cin >> c;
|
2020-06-20 00:04:56 +08:00
|
|
|
switch (c) {
|
2020-07-24 05:25:44 +08:00
|
|
|
case 1:
|
2021-08-16 00:23:18 +08:00
|
|
|
std::cout << "Enter element to add = " << std::endl;
|
2020-07-24 05:25:44 +08:00
|
|
|
std::cin >> x;
|
|
|
|
h = mychain.hash(x);
|
2020-07-24 05:28:45 +08:00
|
|
|
h = std::abs(h);
|
2020-07-24 05:25:44 +08:00
|
|
|
mychain.add(x, h);
|
|
|
|
break;
|
|
|
|
case 2:
|
2021-08-16 00:23:18 +08:00
|
|
|
std::cout << "Enter element to search = " << std::endl;
|
2020-07-24 05:25:44 +08:00
|
|
|
std::cin >> x;
|
|
|
|
h = mychain.hash(x);
|
|
|
|
mychain.find(x, h);
|
|
|
|
break;
|
|
|
|
case 3:
|
2021-08-16 00:23:18 +08:00
|
|
|
std::cout << "Enter element to generate hash = " << std::endl;
|
2020-07-24 05:25:44 +08:00
|
|
|
std::cin >> x;
|
2021-08-18 01:25:08 +08:00
|
|
|
std::cout << "Hash of " << x << " is = " << mychain.hash(x)
|
|
|
|
<< std::endl;
|
2020-07-24 05:25:44 +08:00
|
|
|
break;
|
|
|
|
case 4:
|
|
|
|
mychain.display();
|
|
|
|
break;
|
|
|
|
default:
|
|
|
|
loop = false;
|
|
|
|
break;
|
2020-06-20 00:04:56 +08:00
|
|
|
}
|
2020-07-24 05:25:44 +08:00
|
|
|
std::cout << std::endl;
|
2020-06-20 00:04:56 +08:00
|
|
|
}
|
|
|
|
/*add(1,&head1);
|
|
|
|
add(2,&head1);
|
|
|
|
add(3,&head2);
|
|
|
|
add(5,&head1);
|
|
|
|
display(&head1);
|
|
|
|
display(&head2);*/
|
|
|
|
return 0;
|
2020-07-31 06:39:24 +08:00
|
|
|
}
|