forked from organicmaps/organicmaps
Stage 1. Add NonIntersectingIntervals structure to base.
This commit is contained in:
parent
728aee7180
commit
ffc2dfbdf2
2 changed files with 77 additions and 0 deletions
|
@ -50,6 +50,7 @@ set(
|
|||
mutex.hpp
|
||||
newtype.hpp
|
||||
normalize_unicode.cpp
|
||||
non_intersecting_intervals.hpp
|
||||
observer_list.hpp
|
||||
pprof.cpp
|
||||
pprof.hpp
|
||||
|
|
76
base/non_intersecting_intervals.hpp
Normal file
76
base/non_intersecting_intervals.hpp
Normal file
|
@ -0,0 +1,76 @@
|
|||
#pragma once
|
||||
|
||||
#include "base/assert.hpp"
|
||||
|
||||
#include <algorithm>
|
||||
#include <iterator>
|
||||
#include <set>
|
||||
|
||||
namespace base
|
||||
{
|
||||
///\brief Data structure that maintains a set of non-intersecting intervals.
|
||||
/// A new interval may only be added to the set if it does not intersect any
|
||||
/// of the present intervals, thus the choice which intervals to keep is made
|
||||
/// in a greedy online fashion with no lookahead.
|
||||
template <typename T>
|
||||
class NonIntersectingIntervals
|
||||
{
|
||||
public:
|
||||
/// \brief Adds new interval to set if it doesn't intersect with any that has been already added.
|
||||
/// \return true if there are no such intervals, that intersect the [left, right] interval.
|
||||
bool AddInterval(T left, T right);
|
||||
|
||||
private:
|
||||
struct Interval
|
||||
{
|
||||
Interval(T left, T right);
|
||||
|
||||
struct LessByLeftEnd
|
||||
{
|
||||
bool operator()(Interval const & lhs, Interval const & rhs) const;
|
||||
};
|
||||
|
||||
bool Intersects(Interval const & rhs) const;
|
||||
|
||||
T m_left{};
|
||||
T m_right{};
|
||||
};
|
||||
|
||||
std::set<Interval, typename Interval::LessByLeftEnd> m_leftEnds;
|
||||
};
|
||||
|
||||
template <typename T>
|
||||
bool NonIntersectingIntervals<T>::AddInterval(T left, T right)
|
||||
{
|
||||
Interval interval(left, right);
|
||||
auto it = m_leftEnds.lower_bound(interval);
|
||||
if (it != m_leftEnds.end() && interval.Intersects(*it))
|
||||
return false;
|
||||
|
||||
if (it != m_leftEnds.begin() && interval.Intersects(*std::prev(it)))
|
||||
return false;
|
||||
|
||||
m_leftEnds.emplace(left, right);
|
||||
return true;
|
||||
}
|
||||
|
||||
template <typename T>
|
||||
NonIntersectingIntervals<T>::Interval::Interval(T left, T right)
|
||||
: m_left(left), m_right(right)
|
||||
{
|
||||
CHECK_LESS_OR_EQUAL(left, right, ());
|
||||
}
|
||||
|
||||
template <typename T>
|
||||
bool NonIntersectingIntervals<T>::Interval::LessByLeftEnd::operator()(Interval const & lhs,
|
||||
Interval const & rhs) const
|
||||
{
|
||||
return lhs.m_left < rhs.m_left;
|
||||
};
|
||||
|
||||
template <typename T>
|
||||
bool NonIntersectingIntervals<T>::Interval::Intersects(Interval const & rhs) const
|
||||
{
|
||||
return std::max(m_left, rhs.m_left) <= std::min(m_right, rhs.m_right);
|
||||
}
|
||||
} // namespace coding
|
Loading…
Add table
Reference in a new issue