gpt4 book ai didi

c++ - 为什么我们不能使用 `std::multiset` 和自定义比较 lambda 作为 `std::map` 的值?

转载 作者:可可西里 更新时间:2023-11-01 17:34:17 26 4
gpt4 key购买 nike

这是被问到的后续问题 How to provide custom comparator for `std::multiset` without overloading `operator()`, `std::less`, `std::greater`?

我已经尝试通过以下方式解决。

基本

可以为 std::multiset 提供自定义比较 lambda 函数(自 起)一个类的成员如下:

#include <iostream>
#include <set>

const auto compare = [](int lhs, int rhs) noexcept { return lhs > rhs; };
struct Test
{
std::multiset<int, decltype(compare)> _set{compare};
Test() = default;
};

足够简单。

我的情况

Test的成员类是

std::map<std::string, std::multiset<int, /* custom compare */>> scripts{};

我尝试使用 std::multiset与自定义

  • 仿函数 Compare (案例 - 1)
  • std::greater<> (案例 - 2)
  • lambda 函数(案例 - 3)

前两个选项都是成功的。但是 lambda 作为自定义比较函数的情况它不起作用。这是 MCVC:https://godbolt.org/z/mSHi1p

#include <iostream>
#include <functional>
#include <string>
#include <map>
#include <set>

const auto compare = [](int lhs, int rhs) noexcept { return lhs > rhs; };
class Test
{
private:
struct Compare
{
bool operator()(const int lhs, const int rhs) const noexcept { return lhs > rhs; }
};

private:
// std::multiset<int, Compare> dummy; // works fine
// std::multiset<int, std::greater<>> dummy; // works fine
std::multiset<int, decltype(compare)> dummy{ compare }; // does not work
using CustomMultiList = decltype(dummy);

public:
std::map<std::string, CustomMultiList> scripts{};
};

int main()
{
Test t{};
t.scripts["Linux"].insert(5);
t.scripts["Linux"].insert(8);
t.scripts["Linux"].insert(0);

for (auto a : t.scripts["Linux"]) {
std::cout << a << '\n';
}
}

错误信息:

error C2280 : '<lambda_778ad726092eb2ad4bce2e3abb93017f>::<lambda_778ad726092eb2ad4bce2e3abb93017f>(void)' : attempting to reference a deleted function
note: see declaration of '<lambda_778ad726092eb2ad4bce2e3abb93017f>::<lambda_778ad726092eb2ad4bce2e3abb93017f>'
note: '<lambda_778ad726092eb2ad4bce2e3abb93017f>::<lambda_778ad726092eb2ad4bce2e3abb93017f>(void)' : function was explicitly deleted
note: while compiling class template member function 'std::multiset<int,const <lambda_778ad726092eb2ad4bce2e3abb93017f>,std::allocator<int>>::multiset(void)'
note: see reference to function template instantiation 'std::multiset<int,const <lambda_778ad726092eb2ad4bce2e3abb93017f>,std::allocator<int>>::multiset(void)' being compiled
note: see reference to class template instantiation 'std::multiset<int,const <lambda_778ad726092eb2ad4bce2e3abb93017f>,std::allocator<int>>' being compiled

听起来我试图默认构造传递的 lambda,which is not possible直到 .

如果是这样它发生在哪里是否可以在 范围内使用 lambda 比较函数解决此问题直到

最佳答案

It sounds like I tried to default construct the passed lambda, which is not possible until . If that the case where has it happened?

。这正是由于 std::map::operator[] 的调用而在这里发生的事情在行

t.scripts["Linux"].insert(5);
// ^^^^^^^^^

让我们来看看细节。上面的调用将导致调用以下重载,因为 key 是临时的 std::stringconst char* 构建.

T& operator[]( Key&& key );

C++17 this is equivalent to :

return this->try_emplace(
std::move(key)).first -> second;
// key_type mapped_type
// ^^^^^^^^ ^^^^^^^^^^^
// | |
// | |
// (std::string) (std::multiset<int, decltype(compare)>)
// | ^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^
// | | (default-construction meaning)
// | default-construction --> std::multiset<int, decltype(compare)>{}
// move-construction ^^

key_type(即从 std::string 临时构建的 const char*)应该是 move constructible ,碰巧很好。

mapped_type(即 std::multiset<int, decltype(compare)> )应该是 default construct 首先编辑,这要求比较 lambda 也应该是默认构造的。来自 cppreference.com :

ClosureType::ClosureType()

ClosureType() = delete;   (until C++14)
ClosureType() = default; (since C++20)(only if no captures are specified)

Closure types are not DefaultConstructible. Closure types have a deleted (until C++14)no (since C++14) default constructor. (until C++20)


If no captures are specified, the closure type has a defaulted default constructor. Otherwise, it has no default constructor (this includes the case when there is a capture-default, even if it does not actually capture anything). (since C++20)

这意味着,lambda 闭包类型的默认构造在 C++17 中不可用(这就是编译器错误所提示的)。

另一方面,compare 中没有指定捕获(即无状态 lambda) lambda 在那里,因此它可以被支持 C++20 标准的编译器显式默认。


Is it possible to solve this using a lambda compare function within the scope of till ?

不是通过使用 std::map::operator[] (至于上面解释的原因),但是是的,@JohnZwinck's 在他的回答中提到的方式。我想解释一下它是如何工作的。

std::multiset 的构造函数之一1提供传递比较器对象的可能性。

template< class InputIt >
multiset( InputIt first, InputIt last,
const Compare& comp = Compare(),
// ^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^
const Allocator& alloc = Allocator() );

同时lambda闭包类型的复制构造函数和移动构造函数已经defaulted since C++14 .这意味着,如果我们有可能提供 lambda 作为第一个参数2(通过复制或移动它),它将是Basic 案例,问题中显示了什么。

std::multiset<int, decltype(compare)> dummy{ compare };            // copying
std::multiset<int, decltype(compare)> dummy{ std::move(compare) }; // moving

幸运的是,C++17 引入了成员函数 std::map::try_emplace

template <class... Args>
pair<iterator, bool> try_emplace(key_type&& k, Args&&... args);

通过它可以将 lambda 传递给上述构造函数1 of std::multiset作为第一个参数2,如上所示。如果我们将其变形为 Test 的成员函数类,元素可以插入到 CustomMultiList scripts 的(即值) map 。

解决方案看起来像(与链接的帖子相同,因为我在问这个问题后写了那个答案!)

( See Live )

#include <iostream>
#include <string>
#include <map>
#include <set>

// provide a lambda compare
const auto compare = [](int lhs, int rhs) noexcept { return lhs > rhs; };

class Test
{
private:
// make a std::multi set with custom compare function
std::multiset<int, decltype(compare)> dummy{ compare };
using CustomMultiList = decltype(dummy); // use the type for values of the map
public:
std::map<std::string, CustomMultiList> scripts{};
// warper method to insert the `std::multilist` entries to the corresponding keys
void emplace(const std::string& key, const int listEntry)
{
scripts.try_emplace(key, compare).first->second.emplace(listEntry);
}
// getter function for custom `std::multilist`
const CustomMultiList& getValueOf(const std::string& key) const noexcept
{
static CustomMultiList defaultEmptyList{ compare };
const auto iter = scripts.find(key);
return iter != scripts.cend() ? iter->second : defaultEmptyList;
}
};

int main()
{
Test t{};
// 1: insert using using wrapper emplace method
t.emplace(std::string{ "Linux" }, 5);
t.emplace(std::string{ "Linux" }, 8);
t.emplace(std::string{ "Linux" }, 0);


for (const auto a : t.getValueOf(std::string{ "Linux" }))
{
std::cout << a << '\n';
}
// 2: insert the `CustomMultiList` directly using `std::map::emplace`
std::multiset<int, decltype(compare)> valueSet{ compare };
valueSet.insert(1);
valueSet.insert(8);
valueSet.insert(5);
t.scripts.emplace(std::string{ "key2" }, valueSet);

// 3: since C++20 : use with std::map::operator[]
// latest version of GCC has already included this change
//t.scripts["Linux"].insert(5);
//t.scripts["Linux"].insert(8);
//t.scripts["Linux"].insert(0);

return 0;
}

关于c++ - 为什么我们不能使用 `std::multiset` 和自定义比较 lambda 作为 `std::map` 的值?,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/56410886/

26 4 0
Copyright 2021 - 2024 cfsdn All Rights Reserved 蜀ICP备2022000587号
广告合作:1813099741@qq.com 6ren.com