如果我试图在int
中保存std::thread
和std::unordered_map<int, std::thread>
,一切似乎都能正常工作。
但是,如果我包装std::unordered_map
,那么我在标准库中会出错。
错误是:no matching constructor for initialization of '_Mypair'
有没有办法解决这个问题,并在它被包裹时使其工作?
这有效:
Source.cpp
#include <thread>
#include <unordered_map>
void display() {}
int main()
{
std::unordered_map<int, std::thread> map_;
std::thread tempThread(&display);
map_.emplace(0, std::move(tempThread));
return 0;
}
这不起作用:
MapperWrapper.h
#ifndef MAPPERWRAPPER_H
#define MAPPERWRAPPER_H
#include <unordered_map>
#include <utility>
template <typename KeyType, typename valueType> class MapperWrapper
{
public:
void add(KeyType key, valueType value)
{
mapper.emplace(std::make_pair(key, value));
}
protected:
std::unordered_map<KeyType, valueType> mapper;
};
#endif // MAPPERWRAPPER_H
Source.cpp
#include <thread>
#include "MapperWrapper.h"
void display() {}
int main()
{
MapperWrapper<int, std::thread> map_;
std::thread tempThread(&display);
map_.add(0, std::move(tempThread));
return 0;
}
这会生成错误:
||=== Build: Debug in Testing (compiler: LLVM Clang Compiler) ===|
C:\Program Files (x86)\Microsoft Visual Studio 14.0\VC\include\utility|327|error: no matching constructor for initialization of '_Mypair' (aka 'pair<int, std::thread>')|
C:\Users\usr\Desktop\Project2\Project2\MapperWrapper.h|12|in instantiation of function template specialization 'std::make_pair<int &, std::thread &>' requested here|
C:\Users\usr\Desktop\Project2\Project2\Source.cpp|12|in instantiation of member function 'MapperWrapper<int, std::thread>::add' requested here|
C:\Program Files (x86)\Microsoft Visual Studio 14.0\VC\include\utility|96|note: candidate template ignored: requirement 'is_copy_constructible<thread>::value' was not satisfied [with _Uty1 = int, _Uty2 = std::thread]|
C:\Program Files (x86)\Microsoft Visual Studio 14.0\VC\include\utility|107|note: candidate template ignored: requirement 'is_copy_constructible<thread>::value' was not satisfied [with _Uty1 = int, _Uty2 = std::thread]|
C:\Program Files (x86)\Microsoft Visual Studio 14.0\VC\include\utility|167|note: candidate template ignored: requirement 'is_constructible<thread, thread &>::value' was not satisfied [with _Other1 = int &, _Other2 = std::thread &]|
C:\Program Files (x86)\Microsoft Visual Studio 14.0\VC\include\utility|181|note: candidate template ignored: requirement 'is_constructible<thread, thread &>::value' was not satisfied [with _Other1 = int &, _Other2 = std::thread &]|
C:\Program Files (x86)\Microsoft Visual Studio 14.0\VC\include\utility|85|note: candidate constructor template not viable: requires 0 arguments, but 2 were provided|
C:\Program Files (x86)\Microsoft Visual Studio 14.0\VC\include\utility|121|note: candidate constructor template not viable: requires single argument '_Right', but 2 arguments were provided|
C:\Program Files (x86)\Microsoft Visual Studio 14.0\VC\include\utility|132|note: candidate constructor template not viable: requires single argument '_Right', but 2 arguments were provided|
C:\Program Files (x86)\Microsoft Visual Studio 14.0\VC\include\utility|150|note: candidate constructor template not viable: requires 4 arguments, but 2 were provided|
C:\Program Files (x86)\Microsoft Visual Studio 14.0\VC\include\utility|157|note: candidate constructor template not viable: requires 3 arguments, but 2 were provided|
C:\Program Files (x86)\Microsoft Visual Studio 14.0\VC\include\utility|195|note: candidate constructor template not viable: requires single argument '_Right', but 2 arguments were provided|
C:\Program Files (x86)\Microsoft Visual Studio 14.0\VC\include\utility|209|note: candidate constructor template not viable: requires single argument '_Right', but 2 arguments were provided|
C:\Program Files (x86)\Microsoft Visual Studio 14.0\VC\include\utility|112|note: candidate constructor not viable: requires 1 argument, but 2 were provided|
C:\Program Files (x86)\Microsoft Visual Studio 14.0\VC\include\utility|113|note: candidate constructor not viable: requires 1 argument, but 2 were provided|
||=== Build failed: 1 error(s), 0 warning(s) (0 minute(s), 0 second(s)) ===|
您已经按值接受值类型。由于函数参数是本地的,您可以移动它(也可以移动键)
void add(KeyType key, valueType value)
{
mapper.emplace(std::move(key), std::move(value));
}
也没有必要使用std::make_pair
进入地图。
如果你想要更复杂,你可以为add
添加一些完美的转发支持,以避免创建多余的中间对象。
template<typename... Args>
void add(Args&&... args)
{
mapper.emplace(std::forward<Args>(args)...);
}