[generator] Added ClustersFinder.

This commit is contained in:
Maksim Andrianov 2019-08-29 19:34:51 +03:00 committed by Aleksey Belousov
parent 175ff54ffb
commit 3b26b2e7cb
3 changed files with 105 additions and 32 deletions

View file

@ -235,7 +235,7 @@ private:
{
for (auto const & id : fbWithIds.second)
{
if (cities.count(id) != 0)
if (cities.count(id) == 0)
continue;
auto static const kPromoType = classif().GetTypeByPath({"sponsored", "promo_catalog"});

View file

@ -81,19 +81,15 @@ bool IsTheSamePlace(T const & left, T const & right)
return dist <= GetThresholdM(localityL);
}
template <typename Iterator>
Iterator FindGroupOfTheSamePlaces(Iterator start, Iterator end)
std::vector<std::vector<FeaturePlace>> FindClusters(std::vector<FeaturePlace> && places)
{
CHECK(start != end, ());
auto next = start;
while (++next != end)
{
if (!IsTheSamePlace(*next, *start))
return next;
start = next;
}
return end;
auto func = [](FeaturePlace const & fp) {
auto const & localityChecker = ftypes::IsLocalityChecker::Instance();
auto const locality = localityChecker.GetType(GetPlaceType(fp.GetFb()));
return GetThresholdM(locality);
};
ClustersFinder<FeaturePlace, std::vector> f(std::move(places), func, IsTheSamePlace<FeaturePlace>);
return f.Find();
}
} // namespace
@ -131,6 +127,11 @@ m2::RectD const & FeaturePlace::GetLimitRect() const
return m_limitRect;
}
base::GeoObjectId FeaturePlace::GetMostGenericOsmId() const
{
return GetFb().GetMostGenericOsmId();
}
uint8_t FeaturePlace::GetRank() const
{
return GetFb().GetRank();
@ -194,19 +195,11 @@ std::vector<PlaceProcessor::PlaceWithIds> PlaceProcessor::ProcessPlaces()
places.reserve(nameToGeoObjectIdToFeaturePlaces.second.size());
for (auto const & geoObjectIdToFeaturePlaces : nameToGeoObjectIdToFeaturePlaces.second)
places.emplace_back(geoObjectIdToFeaturePlaces.second);
// Firstly, objects are geometrically sorted.
std::sort(std::begin(places), std::end(places), [](auto const & l, auto const & r) {
auto const & rectL = l.GetLimitRect();
auto const & rectR = r.GetLimitRect();
return rectL.Center() < rectR.Center();
});
// Secondly, a group of similar places is searched for, then a better place is searched in
// this group.
auto start = std::begin(places);
while (start != std::cend(places))
auto const clusters = FindClusters(std::move(places));
for (auto const & cluster : clusters)
{
auto end = FindGroupOfTheSamePlaces(start, std::end(places));
auto best = std::max_element(start, end, IsWorsePlace<FeaturePlace>);
auto best = std::max_element(std::cbegin(cluster), std::cend(cluster), IsWorsePlace<FeaturePlace>);
auto bestFb = best->GetFb();
auto const & localityChecker = ftypes::IsLocalityChecker::Instance();
if (bestFb.IsArea() && localityChecker.GetType(GetPlaceType(bestFb)) != ftypes::NONE)
@ -215,17 +208,15 @@ std::vector<PlaceProcessor::PlaceWithIds> PlaceProcessor::ProcessPlaces()
TransformAreaToPoint(bestFb);
}
std::vector<base::GeoObjectId> ids;
ids.reserve(static_cast<size_t>(std::distance(start, end)));
std::transform(start, end, std::back_inserter(ids), [](auto const & place) {
return place.GetMostGenericOsmId();
});
ids.reserve(cluster.size());
std::transform(std::cbegin(cluster), std::cend(cluster), std::back_inserter(ids),
[](auto const & place) { return place.GetMostGenericOsmId(); });
finalPlaces.emplace_back(std::move(bestFb), std::move(ids));
if (m_boundariesTable)
FillTable(start, end, best);
start = end;
FillTable(std::cbegin(cluster), std::cend(cluster), best);
}
}
return finalPlaces;
}

View file

@ -3,15 +3,96 @@
#include "generator/cities_boundaries_builder.hpp"
#include "generator/feature_builder.hpp"
#include "geometry/mercator.hpp"
#include "geometry/tree4d.hpp"
#include "base/geo_object_id.hpp"
#include <functional>
#include <memory>
#include <queue>
#include <set>
#include <string>
#include <unordered_map>
#include <vector>
namespace generator
{
template <typename T, template<typename, typename> class Container, typename Alloc = std::allocator<T>>
class ClustersFinder
{
public:
using DistanceFunc = std::function<double(T const &)>;
using IsSameFunc = std::function<bool(T const &, T const &)>;
using ConstIterator = T const *;
ClustersFinder(Container<T, Alloc> && container, DistanceFunc distanceFunc, IsSameFunc isSameFunc)
: m_container(std::move(container)), m_distanseFunc(distanceFunc), m_isSameFunc(isSameFunc)
{
for (auto const & e : m_container)
m_tree.Add(&e);
}
std::vector<std::vector<T>> Find()
{
std::vector<std::vector<T>> clusters;
std::set<ConstIterator> unviewed;
for (auto const & e : m_container)
unviewed.insert(&e);
while (!unviewed.empty())
clusters.emplace_back(FindOneCluster(*std::cbegin(unviewed), unviewed));
return clusters;
}
private:
struct TraitsDef
{
m2::RectD const LimitRect(ConstIterator const & it) const { return it->GetLimitRect(); }
};
std::vector<T> FindOneCluster(ConstIterator const & it, std::set<ConstIterator> & unviewed)
{
std::vector<T> cluster{*it};
std::queue<ConstIterator> queue;
queue.emplace(it);
unviewed.erase(it);
while (!queue.empty())
{
auto const current = queue.front();
queue.pop();
auto const queryBbox = GetBboxFor(current);
m_tree.ForEachInRect(queryBbox, [&](auto const & conditate) {
if (current == conditate || unviewed.count(conditate) == 0 || !m_isSameFunc(*current, *conditate))
return;
unviewed.erase(conditate);
queue.emplace(conditate);
cluster.emplace_back(*conditate);
});
}
return cluster;
}
m2::RectD GetBboxFor(ConstIterator const & it)
{
m2::RectD bbox;
it->GetLimitRect().ForEachCorner([&](auto const & p) {
auto const dist = m_distanseFunc(*it);
bbox.Add(MercatorBounds::RectByCenterXYAndSizeInMeters(p, dist));
});
return bbox;
}
Container<T, Alloc> m_container;
DistanceFunc m_distanseFunc;
IsSameFunc m_isSameFunc;
m4::Tree<ConstIterator, TraitsDef> m_tree;
};
bool NeedProcessPlace(feature::FeatureBuilder const & fb);
// This structure encapsulates work with elements of different types.
@ -25,6 +106,7 @@ public:
feature::FeatureBuilder const & GetFb() const;
FeaturesBuilders const & GetFbs() const;
m2::RectD const & GetLimitRect() const;
base::GeoObjectId GetMostGenericOsmId() const;
uint8_t GetRank() const;
std::string GetName() const;
m2::PointD GetKeyPoint() const;