2020-10-17 00:38:00 +08:00
|
|
|
/**
|
|
|
|
* @file sparse_table.cpp
|
2023-01-25 04:03:06 +08:00
|
|
|
* @brief Implementation of [Sparse
|
|
|
|
* Table](https://en.wikipedia.org/wiki/Range_minimum_query) data structure
|
2020-10-17 00:38:00 +08:00
|
|
|
*
|
|
|
|
* @details
|
|
|
|
* Sparse Table is a data structure, that allows answering range queries.
|
2023-01-25 04:03:06 +08:00
|
|
|
* It can answer most range queries in O(logn), but its true power is answering
|
|
|
|
* range minimum queries or equivalent range maximum queries). For those queries
|
|
|
|
* it can compute the answer in O(1) time.
|
2020-10-17 00:38:00 +08:00
|
|
|
*
|
|
|
|
* * Running Time Complexity \n
|
|
|
|
* * Build : O(NlogN) \n
|
|
|
|
* * Range Query : O(1) \n
|
2023-01-25 04:03:06 +08:00
|
|
|
*/
|
2020-10-17 00:38:00 +08:00
|
|
|
|
2023-01-25 04:03:06 +08:00
|
|
|
#include <algorithm>
|
2020-10-17 00:38:00 +08:00
|
|
|
#include <cassert>
|
|
|
|
#include <iostream>
|
2023-01-25 04:03:06 +08:00
|
|
|
#include <vector>
|
2020-10-17 00:38:00 +08:00
|
|
|
|
|
|
|
/**
|
|
|
|
* @namespace range_queries
|
|
|
|
* @brief Range Queries algorithms
|
|
|
|
*/
|
|
|
|
namespace range_queries {
|
|
|
|
/**
|
|
|
|
* @namespace sparse_table
|
|
|
|
* @brief Range queries using sparse-tables
|
|
|
|
*/
|
2023-01-25 04:03:06 +08:00
|
|
|
namespace sparse_table {
|
2020-10-17 00:38:00 +08:00
|
|
|
/**
|
|
|
|
* This function precomputes intial log table for further use.
|
|
|
|
* @param n value of the size of the input array
|
|
|
|
* @return corresponding vector of the log table
|
|
|
|
*/
|
2023-01-25 04:03:06 +08:00
|
|
|
template <typename T>
|
2020-10-17 00:38:00 +08:00
|
|
|
std::vector<T> computeLogs(const std::vector<T>& A) {
|
|
|
|
int n = A.size();
|
|
|
|
std::vector<T> logs(n);
|
|
|
|
logs[1] = 0;
|
2023-01-25 04:03:06 +08:00
|
|
|
for (int i = 2; i < n; i++) {
|
|
|
|
logs[i] = logs[i / 2] + 1;
|
2020-10-17 00:38:00 +08:00
|
|
|
}
|
|
|
|
return logs;
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* This functions builds the primary data structure sparse table
|
|
|
|
* @param n value of the size of the input array
|
|
|
|
* @param A array of the input integers
|
|
|
|
* @param logs array of the log table
|
|
|
|
* @return created sparse table data structure
|
|
|
|
*/
|
2023-01-25 04:03:06 +08:00
|
|
|
template <typename T>
|
|
|
|
std::vector<std::vector<T> > buildTable(const std::vector<T>& A,
|
|
|
|
const std::vector<T>& logs) {
|
2020-10-17 00:38:00 +08:00
|
|
|
int n = A.size();
|
2023-01-25 04:03:06 +08:00
|
|
|
std::vector<std::vector<T> > table(20, std::vector<T>(n + 5, 0));
|
2020-10-17 00:38:00 +08:00
|
|
|
int curLen = 0;
|
2023-01-25 04:03:06 +08:00
|
|
|
for (int i = 0; i <= logs[n]; i++) {
|
2020-10-17 00:38:00 +08:00
|
|
|
curLen = 1 << i;
|
2023-01-25 04:03:06 +08:00
|
|
|
for (int j = 0; j + curLen < n; j++) {
|
2020-10-17 00:38:00 +08:00
|
|
|
if (curLen == 1) {
|
|
|
|
table[i][j] = A[j];
|
2023-01-25 04:03:06 +08:00
|
|
|
} else {
|
|
|
|
table[i][j] =
|
|
|
|
std::min(table[i - 1][j], table[i - 1][j + curLen / 2]);
|
2020-10-17 00:38:00 +08:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return table;
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* This function is the query function to get the range minimum value
|
|
|
|
* @param beg beginning index of the query range
|
|
|
|
* @param end ending index of the query range
|
|
|
|
* @param logs array of the log table
|
|
|
|
* @param table sparse table data structure for the input array
|
|
|
|
* @return minimum value for the [beg, end] range for the input array
|
|
|
|
*/
|
2023-01-25 04:03:06 +08:00
|
|
|
template <typename T>
|
|
|
|
int getMinimum(int beg, int end, const std::vector<T>& logs,
|
|
|
|
const std::vector<std::vector<T> >& table) {
|
2020-10-17 00:38:00 +08:00
|
|
|
int p = logs[end - beg + 1];
|
|
|
|
int pLen = 1 << p;
|
|
|
|
return std::min(table[p][beg], table[p][end - pLen + 1]);
|
|
|
|
}
|
2023-01-25 04:03:06 +08:00
|
|
|
} // namespace sparse_table
|
|
|
|
} // namespace range_queries
|
2020-10-17 00:38:00 +08:00
|
|
|
|
|
|
|
/**
|
|
|
|
* Main function
|
|
|
|
*/
|
|
|
|
int main() {
|
|
|
|
std::vector<int> A{1, 2, 0, 3, 9};
|
|
|
|
std::vector<int> logs = range_queries::sparse_table::computeLogs(A);
|
2023-01-25 04:03:06 +08:00
|
|
|
std::vector<std::vector<int> > table =
|
|
|
|
range_queries::sparse_table::buildTable(A, logs);
|
2020-10-17 00:38:00 +08:00
|
|
|
assert(range_queries::sparse_table::getMinimum(0, 0, logs, table) == 1);
|
|
|
|
assert(range_queries::sparse_table::getMinimum(0, 4, logs, table) == 0);
|
|
|
|
assert(range_queries::sparse_table::getMinimum(2, 4, logs, table) == 0);
|
|
|
|
return 0;
|
|
|
|
}
|