-
Notifications
You must be signed in to change notification settings - Fork 0
/
Copy pathhash_tree.cpp
174 lines (146 loc) · 3.47 KB
/
hash_tree.cpp
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
/*
******************************************************************************
* Copyright (c) 2016 Tencent Inc. All rights reserved
* Author: [email protected]
* Date: 2016-10-30
* Description: 哈希树 - 质数分辨定理实现
******************************************************************************
*/
template <class KeyType, class ValueType, int SIZE = 32>
class HashTree
{
struct Node
{
Node() : occupied_(false), child_({0}) {}
KeyType key_;
ValueType value_;
bool occupied_;
Node *child_[SIZE];
};
public:
HashTree()
{
max_level_ = 0;
prime_ = {0};
for (int i = 2; i <= SIZE; ++i)
{
if (IsPrime(i))
{
prime_[++max_level_] = i;
}
}
}
/**
* 插入函数
* @param key 键
* @param value 值
* @return 成功 true, 失败 false
*/
bool Insert(const KeyType &key, const ValueType &value)
{
return Insert(&root_, 1, key, value);
}
/**
* 查找函数
* @param key 键
* @return 指向value的指针, 找不到时为NULL
*/
const ValueType *Find(const KeyType &key)
{
return Find(&root_, 1, key);
}
/**
* 删除函数
* @param key 键
* @return 任何情况下都返回true
*/
bool Delete(const KeyType &key)
{
return Delete(&root_, 1, key);
}
private:
bool Insert(Node *node, int level, const KeyType &key, const ValueType &value)
{
if (!node->occupied_)
{
node->key_ = key;
node->value_ = value;
node->occupied_ = true;
return true;
}
if (level > max_level_)
{
return false;
}
int idx = key % prime_[level];
if (!node->child_[idx])
{
node->child_[idx] = new Node;
}
return Insert(node->child_[idx], ++level, key, value);
}
const ValueType *Find(Node *node, int level, const KeyType &key)
{
if (node->occupied_)
{
if (node->key_ == key)
{
return &node->value_;
}
}
int idx = key % prime_[level];
if (!node->child_[idx])
{
return 0;
}
return Find(node->child_[idx], ++level, key);
}
bool Delete(Node *node, int level, const KeyType &key)
{
if (node->occupied_)
{
if (node->key_ == key)
{
node->occupied_ = false;
return true;
}
}
int idx = key % prime_[level];
if (!node->child_[idx])
{
return true;
}
return Delete(node->child_[idx], ++level, key);
}
public:
bool IsPrime(int n)
{
for (int i = 2; (i * i) < (n + 1); ++i)
{
if (!(n % i))
{
return false;
}
}
return true;
}
private:
Node root_;
int max_level_;
int prime_[SIZE+1];
};
#include <iostream>
int main()
{
HashTree<int, std::string> tree;
tree.Insert(1111, "AAA");
tree.Insert(2222, "BBB");
const std::string *value;
value = tree.Find(2222);
std::cout << (value ? *value : "NULL") << std::endl;
tree.Delete(2222);
value = tree.Find(2222);
std::cout << (value ? *value : "NULL") << std::endl;
std::cout << tree.IsPrime(3);
return 0;
}