Skip to content

Commit

Permalink
Added EmplaceOrAssign for map.
Browse files Browse the repository at this point in the history
Signed-off-by: Viktor Govako <[email protected]>
  • Loading branch information
vng committed Nov 20, 2024
1 parent a5bb9e1 commit 58c990d
Show file tree
Hide file tree
Showing 2 changed files with 60 additions and 4 deletions.
52 changes: 48 additions & 4 deletions base/base_tests/stl_helpers_tests.cpp
Original file line number Diff line number Diff line change
@@ -1,17 +1,17 @@
#include "testing/testing.hpp"

#include "base/macros.hpp"
#include "base/stl_helpers.hpp"

#include <algorithm>
#include <deque>
#include <utility>
#include <map>
#include <vector>

using namespace base;

namespace stl_helpers_test
namespace stl_helpers_tests
{
using namespace base;

class Int
{
public:
Expand Down Expand Up @@ -326,4 +326,48 @@ UNIT_TEST(AccumulateIntervals)
CheckAccumulateIntervals(idTest, arr1, arr2, res);
}
}

UNIT_TEST(Map_EmplaceOrAssign)
{
{
std::map<std::string, std::string, std::less<>> theMap;

std::string_view key = "key";
std::string_view val1 = "value";
TEST(EmplaceOrAssign(theMap, key, val1).second, ());
TEST_EQUAL(theMap.find(key)->second, val1, ());

std::string_view val2 = "some_long_value";
TEST(!EmplaceOrAssign(theMap, key, val2).second, ());
TEST_EQUAL(theMap.find(key)->second, val2, ());

std::string_view val3 = "some_other_long_value";
TEST(!EmplaceOrAssign(theMap, key, std::string(val3)).second, ());
TEST_EQUAL(theMap.find(key)->second, val3, ());
}

{
class Obj
{
int m_v;

Obj(Obj const &) = delete;
Obj & operator=(Obj const &) = delete;
public:
Obj(int v) : m_v(v) {}
Obj(Obj &&) = default;
Obj & operator=(Obj &&) = default;

bool operator==(Obj const & o) const { return m_v == o.m_v; }
bool operator<(Obj const & o) const { return m_v < o.m_v; }
};

std::map<Obj, Obj> theMap;

TEST(EmplaceOrAssign(theMap, Obj(1), Obj(2)).second, ());
TEST(!EmplaceOrAssign(theMap, Obj(1), Obj(3)).second, ());
TEST(theMap.find(Obj(1))->second == Obj(3), ());
}
}

} // namespace stl_helpers_test
12 changes: 12 additions & 0 deletions base/stl_helpers.hpp
Original file line number Diff line number Diff line change
Expand Up @@ -165,6 +165,18 @@ bool IsExist(Cont const & c, T const & t)
return std::find(std::cbegin(c), end, t) != end;
}

template <class MapT, class K, class V>
auto EmplaceOrAssign(MapT & theMap, K && k, V && v)
{
auto it = theMap.lower_bound(k);
if (it != theMap.end() && k == it->first)
{
it->second = std::forward<V>(v);
return std::make_pair(it, false);
}
return std::make_pair(theMap.emplace_hint(it, std::forward<K>(k), std::forward<V>(v)), true);
}

// Creates a comparer being able to compare two instances of class C
// (given by reference or pointer) by a field or const method of C.
// For example, to create comparer that is able to compare pairs of
Expand Down

0 comments on commit 58c990d

Please sign in to comment.