[indexer] Do not stop ForClosestToPoint at the middle of the central cell processing.

This commit is contained in:
tatiana-yan 2018-10-26 17:42:58 +03:00 committed by mpimenov
parent 492d0b75c0
commit 1a3d0d9c79
2 changed files with 77 additions and 26 deletions

View file

@ -125,11 +125,17 @@ UNIT_TEST(LocalityIndexRankTest)
UNIT_TEST(LocalityIndexTopSizeTest)
{
LocalityObjectVector objects;
objects.m_objects.resize(4);
objects.m_objects[0].SetForTests(1, m2::PointD{1, 0});
objects.m_objects[1].SetForTests(2, m2::PointD{1, 0});
objects.m_objects[2].SetForTests(3, m2::PointD{1, 0});
objects.m_objects[3].SetForTests(4, m2::PointD{1, 0});
objects.m_objects.resize(7);
// Same cell.
objects.m_objects[0].SetForTests(1, m2::PointD{1.0, 0.0});
objects.m_objects[1].SetForTests(2, m2::PointD{1.0, 0.0});
objects.m_objects[2].SetForTests(3, m2::PointD{1.0, 0.0});
objects.m_objects[3].SetForTests(4, m2::PointD{1.0, 0.0});
// Another close cell.
objects.m_objects[4].SetForTests(5, m2::PointD{1.0, 1.0});
objects.m_objects[5].SetForTests(6, m2::PointD{1.0, 1.0});
// Far cell.
objects.m_objects[6].SetForTests(7, m2::PointD{10.0, 10.0});
vector<char> localityIndex;
MemWriter<vector<char>> writer(localityIndex);
@ -137,18 +143,35 @@ UNIT_TEST(LocalityIndexTopSizeTest)
MemReader reader(localityIndex.data(), localityIndex.size());
indexer::GeoObjectsIndex<MemReader> index(reader);
TEST_EQUAL(GetRankedIds(index, m2::PointD{1, 0} /* center */, m2::PointD{0, 0} /* border */,
4 /* topSize */)
TEST_EQUAL(GetRankedIds(index, m2::PointD{1.0, 0.0} /* center */,
m2::PointD{10.0, 10.0} /* border */, 4 /* topSize */)
.size(),
4, ());
TEST_EQUAL(GetRankedIds(index, m2::PointD{1, 0} /* center */, m2::PointD{0, 0} /* border */,
3 /* topSize */)
.size(),
3, ());
TEST_EQUAL(GetRankedIds(index, m2::PointD{4, 0} /* center */, m2::PointD{0, 0} /* border */,
1 /* topSize */)
.size(),
1, ());
}
// 4 objects are indexed at the central cell. Index does not guarantee the order but must
// return 4 objects.
TEST_EQUAL(GetRankedIds(index, m2::PointD{1.0, 0.0} /* center */,
m2::PointD{10.0, 10.0} /* border */, 3 /* topSize */)
.size(),
4, ());
// At the {1.0, 1.0} point there are also 2 objects, but it's not a central cell, index must
// return 5 (topSize) objects.
TEST_EQUAL(GetRankedIds(index, m2::PointD{1.0, 0.0} /* center */,
m2::PointD{10.0, 10.0} /* border */, 5 /* topSize */)
.size(),
5, ());
// The same here. There are not too many objects in central cell. Index must return 5 (topSize)
// objects.
TEST_EQUAL(GetRankedIds(index, m2::PointD{4.0, 0.0} /* center */,
m2::PointD{10.0, 10.0} /* border */, 5 /* topSize */)
.size(),
5, ());
TEST_EQUAL(GetRankedIds(index, m2::PointD{4.0, 0.0} /* center */,
m2::PointD{10.0, 10.0} /* border */, 7 /* topSize */)
.size(),
7, ());
}
} // namespace

View file

@ -52,27 +52,55 @@ public:
}
}
// Applies |processObject| to at most |topSize| object closest to |center| with maximal distance |sizeM|.
// Closest objects are first. Due to perfomance reasons far objects have approximate order.
void ForClosestToPoint(ProcessObject const & processObject, m2::PointD const & center, double sizeM,
uint32_t topSize) const
// Applies |processObject| to the objects located within |radiusM| meters from |center|.
// Application to the closest objects and only to them is not guaranteed and the order
// of the objects is not specified.
// However, the method attempts to process objects that are closer to |center| first
// and stop after |sizeHint| objects have been processed. For stability, if an object from
// an index cell has been processed, all other objects from this cell will be processed too,
// thus probably overflowing the |sizeHint| limit.
void ForClosestToPoint(ProcessObject const & processObject, m2::PointD const & center,
double radiusM, uint32_t sizeHint) const
{
auto const rect =
MercatorBounds::RectByCenterXYAndSizeInMeters(center, sizeM);
MercatorBounds::RectByCenterXYAndSizeInMeters(center, radiusM);
covering::CoveringGetter cov(rect, covering::CoveringMode::Spiral);
covering::Intervals const & intervals = cov.Get<DEPTH_LEVELS>(scales::GetUpperScale());
std::set<uint64_t> objects;
auto process = [topSize, &objects, &processObject](uint64_t storedId) {
if (objects.insert(storedId).second && objects.size() <= topSize)
auto processAll = [&objects, &processObject](uint64_t storedId) {
if (objects.insert(storedId).second)
processObject(LocalityObject::FromStoredId(storedId));
};
auto process = [&](uint64_t storedId) {
if (objects.size() < sizeHint)
processAll(storedId);
};
CHECK_EQUAL(intervals.begin()->first, intervals.begin()->second - 1, ());
auto cellDepth = covering::GetCodingDepth<DEPTH_LEVELS>(scales::GetUpperScale());
auto bestCell = m2::CellId<DEPTH_LEVELS>::FromInt64(intervals.begin()->first, cellDepth);
std::set<int64_t> bestCells;
while (bestCell.Level() > 0)
{
bestCells.insert(bestCell.ToInt64(cellDepth));
bestCell = bestCell.Parent();
--cellDepth;
}
for (auto const & i : intervals)
{
if (objects.size() >= topSize)
return;
m_intervalIndex->ForEach(process, i.first, i.second);
if (bestCells.find(i.first) != bestCells.end())
{
m_intervalIndex->ForEach(processAll, i.first, i.second);
}
else
{
m_intervalIndex->ForEach(process, i.first, i.second);
if (objects.size() >= sizeHint)
return;
}
}
}