feat(splay_tree): add fold and for_each templates for splay_tree and splay_map
Signed-off-by: Leonardo de Moura <leonardo@microsoft.com>
This commit is contained in:
parent
1aca1d2d77
commit
54e63fd4de
4 changed files with 107 additions and 9 deletions
|
@ -5,6 +5,7 @@ Released under Apache 2.0 license as described in the file LICENSE.
|
||||||
Author: Leonardo de Moura
|
Author: Leonardo de Moura
|
||||||
*/
|
*/
|
||||||
#include <iostream>
|
#include <iostream>
|
||||||
|
#include <sstream>
|
||||||
#include "util/test.h"
|
#include "util/test.h"
|
||||||
#include "util/splay_map.h"
|
#include "util/splay_map.h"
|
||||||
#include "util/name.h"
|
#include "util/name.h"
|
||||||
|
@ -31,7 +32,19 @@ static void tst0() {
|
||||||
lean_assert(m2.size() == 3);
|
lean_assert(m2.size() == 3);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
static void tst1() {
|
||||||
|
int2name m;
|
||||||
|
m[10] = name("t1");
|
||||||
|
m[20] = name("t2");
|
||||||
|
lean_assert(fold(m, [](int k, name const &, int a) { return k + a; }, 0) == 30);
|
||||||
|
std::ostringstream out;
|
||||||
|
for_each(m, [&](int, name const & v) { out << v << " "; });
|
||||||
|
std::cout << out.str() << "\n";
|
||||||
|
lean_assert(out.str() == "t1 t2 ");
|
||||||
|
}
|
||||||
|
|
||||||
int main() {
|
int main() {
|
||||||
tst0();
|
tst0();
|
||||||
|
tst1();
|
||||||
return has_violations() ? 1 : 0;
|
return has_violations() ? 1 : 0;
|
||||||
}
|
}
|
||||||
|
|
|
@ -9,6 +9,7 @@ Author: Leonardo de Moura
|
||||||
#include <random>
|
#include <random>
|
||||||
#include <ctime>
|
#include <ctime>
|
||||||
#include <unordered_set>
|
#include <unordered_set>
|
||||||
|
#include <sstream>
|
||||||
#include "util/test.h"
|
#include "util/test.h"
|
||||||
#include "util/splay_tree.h"
|
#include "util/splay_tree.h"
|
||||||
#include "util/timeit.h"
|
#include "util/timeit.h"
|
||||||
|
@ -19,7 +20,7 @@ struct int_lt { int operator()(int i1, int i2) const { return i1 < i2 ? -1 : (i1
|
||||||
typedef splay_tree<int, int_lt> int_splay_tree;
|
typedef splay_tree<int, int_lt> int_splay_tree;
|
||||||
typedef std::unordered_set<int> int_set;
|
typedef std::unordered_set<int> int_set;
|
||||||
|
|
||||||
void tst0() {
|
static void tst0() {
|
||||||
int_splay_tree s;
|
int_splay_tree s;
|
||||||
s.insert(10);
|
s.insert(10);
|
||||||
s.insert(11);
|
s.insert(11);
|
||||||
|
@ -32,7 +33,7 @@ void tst0() {
|
||||||
s.insert(15);
|
s.insert(15);
|
||||||
}
|
}
|
||||||
|
|
||||||
void tst1() {
|
static void tst1() {
|
||||||
int_splay_tree s;
|
int_splay_tree s;
|
||||||
s.insert(10);
|
s.insert(10);
|
||||||
s.insert(3);
|
s.insert(3);
|
||||||
|
@ -73,7 +74,7 @@ void tst1() {
|
||||||
lean_assert(s.empty());
|
lean_assert(s.empty());
|
||||||
}
|
}
|
||||||
|
|
||||||
bool operator==(int_set const & v1, int_splay_tree const & v2) {
|
static bool operator==(int_set const & v1, int_splay_tree const & v2) {
|
||||||
buffer<int> b;
|
buffer<int> b;
|
||||||
// std::cout << v2 << "\n";
|
// std::cout << v2 << "\n";
|
||||||
// std::for_each(v1.begin(), v1.end(), [](int v) { std::cout << v << " "; }); std::cout << "\n";
|
// std::for_each(v1.begin(), v1.end(), [](int v) { std::cout << v << " "; }); std::cout << "\n";
|
||||||
|
@ -141,9 +142,23 @@ static void tst2() {
|
||||||
driver(128, 1000, 10000, 0.5, 0.01);
|
driver(128, 1000, 10000, 0.5, 0.01);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
static void tst3() {
|
||||||
|
int_splay_tree v;
|
||||||
|
v.insert(10);
|
||||||
|
v.insert(5);
|
||||||
|
v.insert(1);
|
||||||
|
v.insert(3);
|
||||||
|
lean_assert_eq(fold(v, [](int a, int b) { return a + b; }, 0), 19);
|
||||||
|
std::ostringstream out;
|
||||||
|
for_each(v, [&](int a) { out << a << " "; });
|
||||||
|
std::cout << out.str() << "\n";
|
||||||
|
lean_assert(out.str() == "1 3 5 10 ");
|
||||||
|
}
|
||||||
|
|
||||||
int main() {
|
int main() {
|
||||||
tst0();
|
tst0();
|
||||||
tst1();
|
tst1();
|
||||||
tst2();
|
tst2();
|
||||||
|
tst3();
|
||||||
return has_violations() ? 1 : 0;
|
return has_violations() ? 1 : 0;
|
||||||
}
|
}
|
||||||
|
|
|
@ -57,5 +57,37 @@ public:
|
||||||
performing an insertion if such key does not already exist.
|
performing an insertion if such key does not already exist.
|
||||||
*/
|
*/
|
||||||
ref operator[](K const & k) { return ref(*this, k); }
|
ref operator[](K const & k) { return ref(*this, k); }
|
||||||
|
|
||||||
|
template<typename F, typename R>
|
||||||
|
R fold(F f, R r) const {
|
||||||
|
auto f_prime = [&](entry const & e, R r) -> R { return f(e.first, e.second, r); };
|
||||||
|
return m_map.fold(f_prime, r);
|
||||||
|
}
|
||||||
|
|
||||||
|
template<typename F>
|
||||||
|
void for_each(F f) const {
|
||||||
|
auto f_prime = [&](entry const & e) { f(e.first, e.second); };
|
||||||
|
return m_map.for_each(f_prime);
|
||||||
|
}
|
||||||
};
|
};
|
||||||
|
template<typename K, typename T, typename CMP>
|
||||||
|
splay_map<K, T, CMP> insert(splay_map<K, T, CMP> const & m, K const & k, T const & v) {
|
||||||
|
auto r = m;
|
||||||
|
r.insert(k, v);
|
||||||
|
return r;
|
||||||
|
}
|
||||||
|
template<typename K, typename T, typename CMP>
|
||||||
|
splay_map<K, T, CMP> erase(splay_map<K, T, CMP> const & m, K const & k) {
|
||||||
|
auto r = m;
|
||||||
|
r.erase(k);
|
||||||
|
return r;
|
||||||
|
}
|
||||||
|
template<typename K, typename T, typename CMP, typename F, typename R>
|
||||||
|
R fold(splay_map<K, T, CMP> const & m, F f, R r) {
|
||||||
|
return m.fold(f, r);
|
||||||
|
}
|
||||||
|
template<typename K, typename T, typename CMP, typename F>
|
||||||
|
void for_each(splay_map<K, T, CMP> const & m, F f) {
|
||||||
|
return m.for_each(f);
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
|
@ -287,11 +287,24 @@ class splay_tree : public CMP {
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
static unsigned size(node const * n) {
|
template<typename F, typename R>
|
||||||
if (n)
|
static R fold(node const * n, F & f, R r) {
|
||||||
return 1 + size(n->m_left) + size(n->m_right);
|
if (n) {
|
||||||
else
|
r = fold(n->m_left, f, r);
|
||||||
return 0;
|
r = f(n->m_value, r);
|
||||||
|
return fold(n->m_right, f, r);
|
||||||
|
} else {
|
||||||
|
return r;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
template<typename F>
|
||||||
|
static void for_each(node const * n, F & f) {
|
||||||
|
if (n) {
|
||||||
|
for_each(n->m_left, f);
|
||||||
|
f(n->m_value);
|
||||||
|
for_each(n->m_right, f);
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
splay_tree(splay_tree const & s, node * new_root):CMP(s), m_ptr(new_root) { node::inc_ref(m_ptr); }
|
splay_tree(splay_tree const & s, node * new_root):CMP(s), m_ptr(new_root) { node::inc_ref(m_ptr); }
|
||||||
|
@ -319,7 +332,7 @@ public:
|
||||||
bool is_eqp(splay_tree const & t) const { return m_ptr == t.m_ptr; }
|
bool is_eqp(splay_tree const & t) const { return m_ptr == t.m_ptr; }
|
||||||
|
|
||||||
/** \brief Return the size of the splay tree */
|
/** \brief Return the size of the splay tree */
|
||||||
unsigned size() const { return size(m_ptr); }
|
unsigned size() const { return fold([](T const &, unsigned a) { return a + 1; }, 0); }
|
||||||
|
|
||||||
/** \brief Insert \c v in this splay tree. */
|
/** \brief Insert \c v in this splay tree. */
|
||||||
void insert(T const & v) {
|
void insert(T const & v) {
|
||||||
|
@ -405,9 +418,34 @@ public:
|
||||||
node::display(out, t.m_ptr);
|
node::display(out, t.m_ptr);
|
||||||
return out;
|
return out;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
\brief Return <tt>f(a_k, ..., f(a_1, f(a_0, r)) ...)</tt>, where
|
||||||
|
<tt>a_0, a_1, ... a_k</tt> are the elements is stored in the splay tree.
|
||||||
|
*/
|
||||||
|
template<typename F, typename R>
|
||||||
|
R fold(F f, R r) const {
|
||||||
|
return fold(m_ptr, f, r);
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
\brief Apply \c f to each value stored in the splay tree.
|
||||||
|
*/
|
||||||
|
template<typename F>
|
||||||
|
void for_each(F f) const {
|
||||||
|
for_each(m_ptr, f);
|
||||||
|
}
|
||||||
};
|
};
|
||||||
template<typename T, typename CMP>
|
template<typename T, typename CMP>
|
||||||
splay_tree<T, CMP> insert(splay_tree<T, CMP> & t, T const & v) { splay_tree<T, CMP> r(t); r.insert(v); return r; }
|
splay_tree<T, CMP> insert(splay_tree<T, CMP> & t, T const & v) { splay_tree<T, CMP> r(t); r.insert(v); return r; }
|
||||||
template<typename T, typename CMP>
|
template<typename T, typename CMP>
|
||||||
splay_tree<T, CMP> erase(splay_tree<T, CMP> & t, T const & v) { splay_tree<T, CMP> r(t); r.erase(v); return r; }
|
splay_tree<T, CMP> erase(splay_tree<T, CMP> & t, T const & v) { splay_tree<T, CMP> r(t); r.erase(v); return r; }
|
||||||
|
template<typename T, typename CMP, typename F, typename R>
|
||||||
|
R fold(splay_tree<T, CMP> const & t, F f, R r) {
|
||||||
|
return t.fold(f, r);
|
||||||
|
}
|
||||||
|
template<typename T, typename CMP, typename F>
|
||||||
|
void for_each(splay_tree<T, CMP> const & t, F f) {
|
||||||
|
return t.for_each(f);
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
Loading…
Reference in a new issue