我正在尝试将回调函数作为函数参数传递。但是在以下代码中获取模板替换失败错误。不确定为什么模板替换失败。
#include<iostream>
#include <map>
#include <tuple>
#include <functional>
template<typename A,typename B>
void myfun(std::map<A,B> & mm, std::function<std::tuple<A,B>(void)> fn)
{
A key;
B val;
std::tie(key,val) = fn();
mm[key] = val;
}
std::tuple<std::string,int> fun()
{
return std::make_tuple(std::string("hi"),1);
}
int main()
{
std::map<std::string,int> gg;
#if 0
//fixed version
std::function<std::tuple<std::string,int>(void)> yy = fun;//fixed
myfun(gg,yy);//fixed
#else
// error causing code
myfun(gg,fun);
#endif
}
错误如下
main.cpp:8:6: note: template argument deduction/substitution failed:
main.cpp:25:17: note: mismatched types 'std::function<std::tuple<_T1, _T2>()>' and 'std::tuple<std::__cxx11::basic_string<char, std::char_traits<char>, std::allocator<char> >, int> (*)()'
myfun(gg,fun);
答案 0 :(得分:5)
编译器不能同时转换为std::function
并推导出模板参数。它不理解任意函数指针和std::function
之间的映射。
这有几种方法。
您可以在呼叫站点明确创建std::function
:
myfun(gg,std::function<std::tuple<std::string,int>(void)>{fun});`
您可以编写make_function
函数来为您推断出类型。您可以在线找到此讨论和实施,例如here,here和here。
myfun(gg,make_function(fun));
您可以忘记std::function
并推断整个函数类型。这是我要采取的方法:
template<typename A,typename B, typename Fun>
void myfun(std::map<A,B> & mm, Fun fn)
{
A key;
B val;
std::tie(key,val) = fn();
mm[key] = val;
}