Added implementation of GpsTrackCollection

This commit is contained in:
Constantin Shalnev 2015-12-04 16:27:13 +03:00
parent 47cb6dfc13
commit 1103562383
3 changed files with 321 additions and 0 deletions

View file

@ -0,0 +1,216 @@
#include "map/gps_track_collection.hpp"
#include "base/assert.hpp"
#include "std/algorithm.hpp"
namespace
{
size_t const kSecondsPerHour = 60 * 60;
size_t const kLinearSearchCount = 10;
// Simple rollbacker which restores deque state
template <typename T>
class Rollbacker
{
public:
Rollbacker(deque<T> & cont)
: m_cont(&cont) , m_size(cont.size())
{}
~Rollbacker()
{
if (m_cont && m_cont->size() > m_size)
m_cont->erase(m_cont->begin() + m_size, m_cont->end());
}
void Reset() { m_cont = nullptr; }
private:
deque<T> * m_cont;
size_t const m_size;
};
} // namespace
GpsTrackCollection::GpsTrackCollection(size_t maxSize, hours duration)
: m_maxSize(maxSize)
, m_duration(duration)
, m_lastId(0)
{
}
size_t GpsTrackCollection::Add(TItem const & item, pair<size_t, size_t> & poppedIds)
{
if (!m_items.empty() && m_items.back().m_timestamp > item.m_timestamp)
{
// Invalid timestamp order
poppedIds = make_pair(kInvalidId, kInvalidId); // Nothing was popped
return kInvalidId; // Nothing was added
}
m_items.emplace_back(item);
++m_lastId;
poppedIds = RemoveExtraItems();
return m_lastId - 1;
}
pair<size_t, size_t> GpsTrackCollection::Add(vector<TItem> const & items, pair<size_t, size_t> & poppedIds)
{
size_t startId = m_lastId;
size_t added = 0;
// Rollbacker ensure strong guarantee if exception happens while adding items
Rollbacker<TItem> rollbacker(m_items);
for (auto const & item : items)
{
if (!m_items.empty() && m_items.back().m_timestamp > item.m_timestamp)
continue;
m_items.emplace_back(item);
++added;
}
rollbacker.Reset();
if (0 == added)
{
// Invalid timestamp order
poppedIds = make_pair(kInvalidId, kInvalidId); // Nothing was popped
return make_pair(kInvalidId, kInvalidId); // Nothing was added
}
m_lastId += added;
poppedIds = RemoveExtraItems();
return make_pair(startId, startId + added - 1);
}
hours GpsTrackCollection::GetDuration() const
{
return m_duration;
}
pair<size_t, size_t> GpsTrackCollection::SetDuration(hours duration)
{
m_duration = duration;
if (m_items.empty())
return make_pair(kInvalidId, kInvalidId);
return RemoveExtraItems();
}
pair<size_t, size_t> GpsTrackCollection::Clear(bool resetIds)
{
if (m_items.empty())
{
if (resetIds)
m_lastId = 0;
return make_pair(kInvalidId, kInvalidId);
}
ASSERT(m_lastId >= m_items.size(), ());
// Range of popped items
auto const res = make_pair(m_lastId - m_items.size(), m_lastId - 1);
// Use move from an empty deque to free memory.
m_items = deque<TItem>();
if (resetIds)
m_lastId = 0;
return res;
}
size_t GpsTrackCollection::GetSize() const
{
return m_items.size();
}
size_t GpsTrackCollection::GetMaxSize() const
{
return m_maxSize;
}
bool GpsTrackCollection::IsEmpty() const
{
return m_items.empty();
}
pair<double, double> GpsTrackCollection::GetTimestampRange() const
{
if (m_items.empty())
return make_pair(0, 0);
ASSERT(m_items.front().m_timestamp <= m_items.back().m_timestamp, ());
return make_pair(m_items.front().m_timestamp, m_items.back().m_timestamp);
}
pair<size_t, size_t> GpsTrackCollection::RemoveUntil(deque<TItem>::iterator i)
{
auto const res = make_pair(m_lastId - m_items.size(),
m_lastId - m_items.size() + distance(m_items.begin(), i) - 1);
m_items.erase(m_items.begin(), i);
return res;
}
pair<size_t, size_t> GpsTrackCollection::RemoveExtraItems()
{
if (m_items.empty())
return make_pair(kInvalidId, kInvalidId); // Nothing to remove
double const lowerBound = m_items.back().m_timestamp - m_duration.count() * kSecondsPerHour;
ASSERT(m_lastId >= m_items.size(), ());
if (m_items.front().m_timestamp >= lowerBound)
{
// All items lie on right side of lower bound,
// but need to remove items by size.
if (m_items.size() <= m_maxSize)
return make_pair(kInvalidId, kInvalidId); // Nothing to remove, all points survived.
return RemoveUntil(m_items.begin() + m_items.size() - m_maxSize);
}
if (m_items.back().m_timestamp <= lowerBound)
{
// All items lie on left side of lower bound. Remove all items.
return RemoveUntil(m_items.end());
}
bool found = false;
auto i = m_items.begin();
// First, try linear search for short distance. It is common case for sliding window
// when new items will pop up old items.
for (size_t j = 0; j < kLinearSearchCount && !found; ++i, ++j)
{
ASSERT(i != m_items.end(), ());
if (i->m_timestamp >= lowerBound)
found = true;
}
// If item wasn't found by linear search, since m_items are sorted by timestamp, use lower_bound to find bound
if (!found)
{
TItem t;
t.m_timestamp = lowerBound;
i = lower_bound(i, m_items.end(), t, [](TItem const & a, TItem const & b)->bool{ return a.m_timestamp < b.m_timestamp; });
ASSERT(i != m_items.end(), ());
}
// If remaining part has size more than max size then cut off to leave max size
size_t const remains = distance(i, m_items.end());
if (remains > m_maxSize)
i += remains - m_maxSize;
return RemoveUntil(i);
}

View file

@ -0,0 +1,103 @@
#pragma once
#include "platform/location.hpp"
#include "std/chrono.hpp"
#include "std/deque.hpp"
#include "std/limits.hpp"
#include "std/utility.hpp"
#include "std/vector.hpp"
class GpsTrackCollection final
{
public:
static size_t const kInvalidId = numeric_limits<size_t>::max();
using TItem = location::GpsTrackInfo;
/// Constructor
/// @param maxSize - max number of items in collection
/// @param duration - duration in hours
GpsTrackCollection(size_t maxSize, hours duration);
/// Adds new point in the collection.
/// @param item - item to be added.
/// @param poppedIds - output, which contains range of identifiers popped items or
/// pair(kInvalidId,kInvalidId) if nothing was removed
/// @returns the item unique identifier or kInvalidId if point has incorrect time.
size_t Add(TItem const & items, pair<size_t, size_t> & poppedIds);
/// Adds set of new points in the collection.
/// @param items - set of items to be added.
/// @param poppedIds - output, which contains range of identifiers popped items or
/// pair(kInvalidId,kInvalidId) if nothing was removed
/// @returns range of identifiers of added items or pair(kInvalidId,kInvalidId) if nothing was added
/// @note items which does not conform to timestamp sequence, is not added.
pair<size_t, size_t> Add(vector<TItem> const & item, pair<size_t, size_t> & poppedIds);
/// Get current duration in hours
/// @returns current duration in hours
hours GetDuration() const;
/// Sets duration in hours.
/// @param duration - new duration value
/// @return range of item identifiers, which were removed or
/// pair(kInvalidId,kInvalidId) if nothing was removed
pair<size_t, size_t> SetDuration(hours duration);
/// Removes all points from the collection.
/// @param resetIds - if it is set to true, then new identifiers will start from 0,
/// otherwise new identifiers will continue from returned value res.second + 1
/// @return range of item identifiers, which were removed or
/// pair(kInvalidId,kInvalidId) if nothing was removed
pair<size_t, size_t> Clear(bool resetIds = true);
/// Returns true if collection is empty, otherwise returns false.
bool IsEmpty() const;
/// Returns number of items in the collection
size_t GetSize() const;
/// Returns range of timestamps of collection, where res.first is lower bound and
/// res.second is upper bound. If collection is empty, then returns pair(0, 0).
pair<double, double> GetTimestampRange() const;
/// Returns max size of collection
size_t GetMaxSize() const;
/// Enumerates items in the collection.
/// @param f - callable object, which is called with params - item and item id,
/// if f returns false, then enumeration is stopped.
/// @param pos - position index to start enumeration
template <typename F>
void ForEach(F && f, size_t pos = 0) const
{
if (pos >= m_items.size())
return;
auto i = m_items.cbegin() + pos, iend = m_items.cend();
size_t id = m_lastId - m_items.size() + pos;
for (; i != iend; ++i, ++id)
{
TItem const & item = *i;
size_t const itemId = id;
if (!f(item, itemId))
break;
}
}
private:
// Removes items in range [m_items.begin(), i) and returnd
// range of identifiers of removed items
pair<size_t, size_t> RemoveUntil(deque<TItem>::iterator i);
// Removes items extra by timestamp and max size
pair<size_t, size_t> RemoveExtraItems();
size_t const m_maxSize;
hours m_duration;
deque<TItem> m_items; // asc. sorted by timestamp
size_t m_lastId;
};

View file

@ -21,6 +21,7 @@ HEADERS += \
framework.hpp \
ge0_parser.hpp \
geourl_process.hpp \
gps_track_collection.hpp \
gps_track_container.hpp \
mwm_url.hpp \
storage_bridge.hpp \
@ -41,6 +42,7 @@ SOURCES += \
framework.cpp \
ge0_parser.cpp \
geourl_process.cpp \
gps_track_collection.cpp \
gps_track_container.cpp \
mwm_url.cpp \
storage_bridge.cpp \