c++ - 模板参数推断失败
问题描述
考虑以下代码:
template<class T>
vector<T> filter(typename vector<T>::iterator begin,
typename vector<T>::iterator end,
bool (*cond)(T a))
{
vector<T> vec;
for (typename vector<T>::iterator it = begin; it != end; it++) {
if (cond(*it)) {
vec.push_back(*it);
}
}
return vec;
}
vector<int> vec = { 1,2,3,4,5,6,7,8,9,10 };
auto another_vec = filter<int>(vec.begin(), vec.end(), [](int a) {return a > 5; });
当我从函数过滤器的调用中删除类型时,代码无法编译,即编写时
filter(vec.begin(), vec.end(), [](int a) {return a > 5; });
我的问题是,为什么?编译器可以从 lambda 和迭代器中推断出类型。
我得到的错误是:
错误 C2784 'std::vector> filter(vector>::iterator,vector>::iterator,bool (__cdecl *)(T))':无法从'main::' 示例 c:\users\danii\documents\visual studio 2017\projects\example\example\source.cpp 24
我找不到有关此问题的详细信息。我的猜测是,编译器不能推断内部类型?(例如,不能从vector推断出int)。如果是这样,为什么会这样?如果不是,是什么情况?有什么办法可以解决吗?
我遇到的另一件事是使用迭代器本身作为模板,即类似
template <class T, class iter, class cond>
vector<T> filter(iter begin, iter end, cond c)
是正确的编程吗?这段代码对我来说有点可疑。
解决方案
这是因为 lambda 类型不完全是预期的函数类型,所以模板推导不起作用。编译器不能同时进行隐式转换和模板推导。如果你使用一个函数,它会:
bool f(int a) {
return a > 5;
}
int main() {
vector<int> vec = { 1,2,3,4,5,6,7,8,9,10 };
auto another_vec = filter(vec.begin(), vec.end(), f);
return 0;
}
推荐阅读
- r - 预测包不适用于 Databricks(R 版 3.5.2)
- java - 我可以在同一个 jvm 上使用 kryo 输出对完全相同的类的实例进行深度比较吗?
- cron - 如何从今天或 X 日期开始每 3 周创建一个 cron 计划
- python - 即使在成功连接并在 kafka 消费者控制台中获取消息后,也无法使用来自 kafka 主题的消息(使用 Python)
- verilog - 关于 Verilog 中的 $monitor,粗体线是什么意思?
- javascript - 如何要求用户打开他们的蓝牙?使用 javascript
- node.js - 没有错误地触发子进程功能
- spring-boot - Gradle5.2 中的 Gradle kotlin 未解决的参考:dependtest
- c# - 值不能为空。参数名称:连接字符串
- python - LookupError:没有安装标签为“管理员”的应用程序。当我尝试使用 Django 运行程序时