diff --git a/lib/graph/priority_queue.hpp b/lib/graph/priority_queue.hpp index ae6496b..9a5123a 100644 --- a/lib/graph/priority_queue.hpp +++ b/lib/graph/priority_queue.hpp @@ -38,10 +38,14 @@ public: void pop() { m_map.erase(m_map.begin()); } void push(const Key& key, const T& value) { m_map.emplace(key, value); } bool modifyKey(const Key& key, const T& value, const Key& new_key) { - auto it = std::find_if(m_map.begin(), m_map.end(), - [&key, &value](const std::pair & p) { return p.first == key && p.second == value; } ); + + const auto eq_it = m_map.equal_range(key); + if (eq_it.first == m_map.end()) return false; + + const auto it = std::find_if(eq_it.first, eq_it.second, [&value](const std::pair & p) { return p.second == value; } ); if (it == m_map.end()) return false; - T v = it->second; // take a copy + + auto v = it->second; // take a copy m_map.erase(it); m_map.emplace(new_key, v); return true; diff --git a/test/graph/test_priority_queue.cpp b/test/graph/test_priority_queue.cpp index 2ebbd56..04c62a2 100644 --- a/test/graph/test_priority_queue.cpp +++ b/test/graph/test_priority_queue.cpp @@ -13,7 +13,7 @@ typedef float2 V; typedef std::pair P; SECTION("empty") { - PriorityQueue pq; + const PriorityQueue pq; REQUIRE( pq.size() == 0 ); REQUIRE( pq.empty() == true ); @@ -27,7 +27,7 @@ typedef std::pair P; REQUIRE( pq.size() == 1 ); REQUIRE( pq.empty() == false ); - P p = pq.top(); + const P p = pq.top(); REQUIRE( p.second == f2 ); pq.pop(); @@ -48,20 +48,17 @@ typedef std::pair P; REQUIRE( pq.size() == 3 ); REQUIRE( pq.empty() == false ); - P p1 = pq.top(); - pq.pop(); + const P p1 = pq.top(); pq.pop(); REQUIRE( p1.first == 1.0f ); REQUIRE( p1.second == f2_1 ); REQUIRE( pq.size() == 2 ); - P p2 = pq.top(); - pq.pop(); + const P p2 = pq.top(); pq.pop(); REQUIRE( p2.first == 2.0f ); REQUIRE( p2.second == f2_2 ); REQUIRE( pq.size() == 1 ); - P p3 = pq.top(); - pq.pop(); + const P p3 = pq.top(); pq.pop(); REQUIRE( p3.first == 3.0f ); REQUIRE( p3.second == f2_3 ); @@ -82,20 +79,101 @@ typedef std::pair P; pq.modifyKey(1.0f, f2_1, 1.0f + 3.0f); pq.modifyKey(3.0f, f2_3, 3.0f - 2.0f); - P p1 = pq.top(); - pq.pop(); + const P p1 = pq.top(); pq.pop(); REQUIRE( p1.first == 1.0f ); REQUIRE( p1.second == f2_3 ); - P p2 = pq.top(); - pq.pop(); + const P p2 = pq.top(); pq.pop(); REQUIRE( p2.first == 2.0f ); REQUIRE( p2.second == f2_2 ); - P p3 = pq.top(); - pq.pop(); + const P p3 = pq.top(); pq.pop(); REQUIRE( p3.first == 4.0f ); REQUIRE( p3.second == f2_1 ); } + SECTION("Same key multivalue") { + PriorityQueue pq; + const V f2_1(1.0f, 1.0f); + const V f2_21(2.0f, 21.0f); + const V f2_22(2.0f, 22.0f); + const V f2_23(2.0f, 23.0f); + const V f2_24(2.0f, 24.0f); + const V f2_34(3.0f, 34.0f); + const V f2_33(3.0f, 32.0f); + + pq.push(3.0f, f2_34); + pq.push(1.0f, f2_1); + pq.push(2.0f, f2_23); + pq.push(3.0f, f2_33); + pq.push(2.0f, f2_21); + pq.push(2.0f, f2_22); + + const P p1 = pq.top(); pq.pop(); + REQUIRE( p1.first == 1.0f ); + REQUIRE( p1.second == f2_1 ); + + // priority queue has a multimap as underlying data_structure: + // "If the container has elements with equivalent key, inserts at the upper bound of that range" + // so with same keys, element pop order is same as adding order + const P p2 = pq.top(); pq.pop(); + REQUIRE( p2.first == 2.0f ); + REQUIRE( p2.second == f2_23 ); + const P p3 = pq.top(); pq.pop(); + REQUIRE( p3.first == 2.0f ); + REQUIRE( p3.second == f2_21 ); + const P p4 = pq.top(); pq.pop(); + REQUIRE( p4.first == 2.0f ); + REQUIRE( p4.second == f2_22 ); + + const P p5 = pq.top(); pq.pop(); + REQUIRE( p5.first == 3.0f ); + REQUIRE( p5.second == f2_34 ); + const P p6 = pq.top(); pq.pop(); + REQUIRE( p6.first == 3.0f ); + REQUIRE( p6.second == f2_33 ); + } + + SECTION("Same key multivalue modyfykey") { + PriorityQueue pq; + const V f2_1(1.0f, 1.0f); + const V f2_21(2.0f, 21.0f); + const V f2_22(2.0f, 22.0f); + const V f2_23(2.0f, 23.0f); + const V f2_24(2.0f, 24.0f); + const V f2_34(3.0f, 34.0f); + const V f2_33(3.0f, 32.0f); + + pq.push(3.0f, f2_34); + pq.push(1.0f, f2_1); + pq.push(2.0f, f2_23); + pq.push(3.0f, f2_33); + pq.push(2.0f, f2_21); + pq.push(2.0f, f2_22); + + pq.modifyKey(2.0f, f2_21, 1.0f); + pq.modifyKey(3.0f, f2_33, 2.0f); + + const P p1 = pq.top(); pq.pop(); + REQUIRE( p1.first == 1.0f ); + REQUIRE( p1.second == f2_1 ); + const P p3 = pq.top(); pq.pop(); + REQUIRE( p3.first == 1.0f ); + REQUIRE( p3.second == f2_21 ); + + const P p2 = pq.top(); pq.pop(); + REQUIRE( p2.first == 2.0f ); + REQUIRE( p2.second == f2_23 ); + const P p4 = pq.top(); pq.pop(); + REQUIRE( p4.first == 2.0f ); + REQUIRE( p4.second == f2_22 ); + const P p6 = pq.top(); pq.pop(); + REQUIRE( p6.first == 2.0f ); + REQUIRE( p6.second == f2_33 ); + + const P p5 = pq.top(); pq.pop(); + REQUIRE( p5.first == 3.0f ); + REQUIRE( p5.second == f2_34 ); + + } }