Made FastSet testable with template magic
parent
0d292e6023
commit
fbc4b42d05
|
|
@ -18,11 +18,25 @@
|
|||
|
||||
#pragma once
|
||||
|
||||
#include <gtsam/base/Testable.h>
|
||||
|
||||
#include <set>
|
||||
#include <iostream>
|
||||
#include <string>
|
||||
#include <cmath>
|
||||
#include <boost/pool/pool_alloc.hpp>
|
||||
#include <boost/mpl/has_xxx.hpp>
|
||||
#include <boost/utility/enable_if.hpp>
|
||||
|
||||
BOOST_MPL_HAS_XXX_TRAIT_DEF(print)
|
||||
|
||||
namespace gtsam {
|
||||
|
||||
// This is used internally to allow this container to be Testable even when it
|
||||
// contains non-testable elements.
|
||||
template<typename VALUE, class ENABLE = void>
|
||||
struct FastSetTestableHelper;
|
||||
|
||||
/**
|
||||
* FastSet is a thin wrapper around std::set that uses the boost
|
||||
* fast_pool_allocator instead of the default STL allocator. This is just a
|
||||
|
|
@ -30,26 +44,94 @@ namespace gtsam {
|
|||
* we've seen that the fast_pool_allocator can lead to speedups of several %.
|
||||
* @ingroup base
|
||||
*/
|
||||
template<typename VALUE>
|
||||
class FastSet : public std::set<VALUE, std::less<VALUE>, boost::fast_pool_allocator<VALUE> > {
|
||||
template<typename VALUE, class ENABLE = void>
|
||||
class FastSet: public std::set<VALUE, std::less<VALUE>, boost::fast_pool_allocator<VALUE> >,
|
||||
public Testable<FastSet<VALUE> > {
|
||||
|
||||
public:
|
||||
|
||||
typedef std::set<VALUE, std::less<VALUE>, boost::fast_pool_allocator<VALUE> > Base;
|
||||
|
||||
/** Default constructor */
|
||||
FastSet() {}
|
||||
FastSet() {
|
||||
}
|
||||
|
||||
/** Constructor from a range, passes through to base class */
|
||||
template<typename INPUTITERATOR>
|
||||
FastSet(INPUTITERATOR first, INPUTITERATOR last) : Base(first, last) {}
|
||||
FastSet(INPUTITERATOR first, INPUTITERATOR last) :
|
||||
Base(first, last) {
|
||||
}
|
||||
|
||||
/** Copy constructor from another FastSet */
|
||||
FastSet(const FastSet<VALUE>& x) : Base(x) {}
|
||||
FastSet(const FastSet<VALUE>& x) :
|
||||
Base(x) {
|
||||
}
|
||||
|
||||
/** Copy constructor from the base map class */
|
||||
FastSet(const Base& x) : Base(x) {}
|
||||
FastSet(const Base& x) :
|
||||
Base(x) {
|
||||
}
|
||||
|
||||
/** Print to implement Testable */
|
||||
void print(const std::string& str = "") const { FastSetTestableHelper<VALUE>::print(*this, str); }
|
||||
|
||||
/** Check for equality within tolerance to implement Testable */
|
||||
bool equals(const FastSet<VALUE>& other, double tol = 1e-9) const { return FastSetTestableHelper<VALUE>::equals(*this, other, tol); }
|
||||
};
|
||||
|
||||
// This is the default Testable interface for *non*Testable elements, which
|
||||
// uses stream operators.
|
||||
template<typename VALUE, class ENABLE>
|
||||
struct FastSetTestableHelper {
|
||||
|
||||
typedef FastSet<VALUE> Set;
|
||||
|
||||
static void print(const Set& set, const std::string& str) {
|
||||
std::cout << str << "\n";
|
||||
for (typename Set::const_iterator it = set.begin(); it != set.end(); ++it)
|
||||
std::cout << " " << *it << "\n";
|
||||
std::cout.flush();
|
||||
}
|
||||
|
||||
static bool equals(const Set& set1, const Set& set2, double tol) {
|
||||
typename Set::const_iterator it1 = set1.begin();
|
||||
typename Set::const_iterator it2 = set2.begin();
|
||||
while (it1 != set1.end()) {
|
||||
if (it2 == set2.end() ||
|
||||
fabs((double)(*it1) - (double)(*it2) > tol))
|
||||
return false;
|
||||
++it1;
|
||||
++it2;
|
||||
}
|
||||
return true;
|
||||
}
|
||||
};
|
||||
|
||||
// This is the Testable interface for Testable elements
|
||||
template<typename VALUE>
|
||||
struct FastSetTestableHelper<VALUE, typename boost::enable_if<has_print<VALUE> >::type> {
|
||||
|
||||
typedef FastSet<VALUE> Set;
|
||||
|
||||
static void print(const Set& set, const std::string& str) {
|
||||
std::cout << str << "\n";
|
||||
for (typename Set::const_iterator it = set.begin(); it != set.end(); ++it)
|
||||
it->print(" ");
|
||||
std::cout.flush();
|
||||
}
|
||||
|
||||
static bool equals(const Set& set1, const Set& set2, double tol) {
|
||||
typename Set::const_iterator it1 = set1.begin();
|
||||
typename Set::const_iterator it2 = set2.begin();
|
||||
while (it1 != set1.end()) {
|
||||
if (it2 == set2.end() ||
|
||||
!it1->equals(*it2, tol))
|
||||
return false;
|
||||
++it1;
|
||||
++it2;
|
||||
}
|
||||
return true;
|
||||
}
|
||||
};
|
||||
|
||||
}
|
||||
|
|
|
|||
Loading…
Reference in New Issue