-
Notifications
You must be signed in to change notification settings - Fork 5
/
Copy pathveboas.cpp
executable file
·145 lines (106 loc) · 3.58 KB
/
veboas.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
/*
A cache-oblivious [0] version of a van Emde Boas search tree [1],
stored implicitly as an array.
0: http://erikdemaine.org/papers/BRICS2002/paper.pdf
1: https://en.wikipedia.org/wiki/Van_Emde_Boas_tree
Thanks to Jeff Erickson for suggesting the problem, and to John
Fischer and Yasutaka Furukawa for pointing out how to encode an
element's position using its bitstring as a traversal code.
Leslie Wu
*/
#include <iostream>
#include <cmath>
#include <vector>
#include <algorithm>
using namespace std;
// Data
const char* veb_str = "HDLBACFEGJIKNMO";
inline int power_of_two(int exponent) { return 1 << exponent; }
// Compute things similar to the hyperfloor and hyperceiling
void hyper_compute(int n, int& d, int& D, int& subtree_size, int& subtree_leaf_count)
{
int h = (int)ceil(log((float)n+1)/log(2.0f));
d = h / 2;
D = power_of_two(d) - 1;
int delta = h - d;
subtree_size = power_of_two(delta) - 1;
subtree_leaf_count = power_of_two(h - 1);
}
// Implicit van Emde Boas binary search
int veb_search(const char* veb_array, int length, char elt)
{
int d, D, subtree_size, subtree_leaf_count;
hyper_compute(length, d, D, subtree_size, subtree_leaf_count);
if (length > 1) {
// Recurse on top half of tree
int subtree_index = veb_search(veb_array, D, elt);
if (subtree_index < 0) return subtree_index;
int offset = subtree_index * subtree_size + D;
// If not in top half, use subtree index to find place in bottom half
int bottom_subtree_index = veb_search(veb_array + offset, subtree_size, elt);
return subtree_leaf_count*subtree_index + bottom_subtree_index;
} else {
char root = veb_array[0];
if (elt == root) {
std::cout << "Found " << elt << " at index "
<< (int)(veb_array - veb_str) << std::endl;
return -1;
}
return (elt < root) ? 0 : 1;
}
}
int tree_size(int depth) { return power_of_two(depth) - 1; }
// Given a sorted index in the range [1, 2^{tree_height}-1], return the VEB address
int veb_index(int n, int tree_height)
{
if (tree_height <= 1) return n;
// Chop n's bitstring into two halves
int bottom_half = tree_height / 2;
int top_half = tree_height - bottom_half;
// Store each half
int top_n = n >> bottom_half;
int bottom_n = n & (power_of_two(bottom_half) - 1);
// Recurse
if (bottom_n == 0) return veb_index(top_n, top_half);
int top_size = power_of_two(top_half) - 1;
int subtree_size = power_of_two(bottom_half) - 1;
int top_address = top_n * tree_size(bottom_half) + tree_size(top_half);
int bot_address = veb_index(bottom_n, bottom_half);
return top_address + bot_address;
}
int main()
{
// Print mapping from sorted index to VEB index
int k = 4;
int pow_k = power_of_two(k);
for (int i=1; i < pow_k; i++) {
int idx = veb_index(i, k);
std::cout << "i = " << i << " -> " << idx << std::endl;
}
// To create implicit VEB structure, generate VEB addresses as satellite data,
// tag, then sort
const char* alpha = "ABCDEFGHIJKLMNO";
int len = strlen(alpha);
vector<pair<int,char> > indices(len);
int power = power_of_two((int)(ceil(log((float)len)/log(2.0))));
int i;
for (i=0; i < len; i++) {
int idx = veb_index(i+1, power);
indices[i] = make_pair(idx, alpha[i]);
}
sort(begin(indices), end(indices));
for (i=0; i < len; i++) {
std::cout << indices[i].second;
}
std::cout << std::endl;
// Search array
int length = (int)strlen(veb_str);
const char* search = "HDLBACFEGJIKNMOabc";
int search_len = (int)strlen(search);
for (int i=0; i < search_len; i++) {
char elt = search[i];
cout << "Looking for " << elt << "..." << endl;
veb_search(veb_str, length, elt);
}
return 0;
}