- html - 出于某种原因,IE8 对我的 Sass 文件中继承的 html5 CSS 不友好?
- JMeter 在响应断言中使用 span 标签的问题
- html - 在 :hover and :active? 上具有不同效果的 CSS 动画
- html - 相对于居中的 html 内容固定的 CSS 重复背景?
我正在尝试将 lambda 传递给采用模板函数类型的模板函数。编译时出现错误 candidate template ignored: could not match...
.
但是,当我尝试将相同的 lambda 传递给采用模板函数类型的模板类时,它会编译并运行。
考虑以下代码 (c++17)
#include <functional>
// define the template function type
template<typename T, typename...S>
using compose_fn_t = std::function<T(S...)>;
// define a template function which accepts the template function type
template<typename T, typename... S>
void compose(compose_fn_t<T, S...> fn) {};
// define a template class which accepts the template function type
template<typename T, typename... S>
class Compose {
public:
Compose(compose_fn_t<T, S...> fn) {};
};
// some arbitrary types
struct A{};
struct B{};
struct C{};
int main() {
compose<A, B, C>
([] (B b, C c) -> A { return {}; }); // this will not compile!
Compose<A, B, C>
([] (B b, C c) -> A { return {}; }); // this compiles and runs correctly!
return 0;
}
当我使用 compose<A, B, C>
编译时,它会抛出以下错误
$ g++ -std=c++17 -o main main.cpp
main.cpp:18:3: error: no matching function for call to 'compose'
compose<A, B, C>
^~~~~~~~~~~~~~~~
main.cpp:8:6: note: candidate template ignored: could not match 'function<A (B, C, type-parameter-0-1...)>' against '(lambda at main.cpp:19:6)'
void compose(compose_fn_t<T, S...> fn) {
^
1 error generated.
这个额外的是什么type-parameter-0-1
模板函数类型 ( compose_fn_t
) 所期望的类型?
最佳答案
如果为函数调用指定模板参数列表,如compose<A, B, C>
,那么如果模板参数多于实参,则该列表被视为部分列表。
调用该函数仍然会对剩余的模板参数进行模板实参推导。
在您的情况下,参数包的其余参数是根据 compose_fn_t
推导出来的。参数(前三个模板参数已经确定),但是失败了,因为 lambda 无法推导出 std::function
类型。
您需要将函数参数中使用的模板参数强制放入非推导上下文中以避免这种情况。一种方法是使用
template<typename T, typename... S>
void compose(typename std::type_identity<compose_fn_t<T, S...>>::type fn) {};
由于范围解析运算符左侧的所有内容 ::
是非推导的。但这也意味着,如果没有模板参数列表,您将无法调用该函数。
std::type_identity
是 C++20 功能,但您可以轻松实现自己的功能。它什么也不做,只是返回 type
中赋予它的类型。成员(member):
template<typename T>
struct type_identity {
using type = T;
};
或者通过转发引用来获取参数并将其转发到 std::function
在函数体中构造,避免任何推导:
template<typename T, typename... S, typename F>
void compose(F&& f) {
compose_fn_t<T, S...> fn{std::forward<F>(f)};
// Use fn as before
};
<小时/>
这不是类模板的问题,因为只有在根本没有提供模板参数列表的情况下才会执行类模板参数推导 (CTAD)。
<小时/>您还可以在 std::function
上使用 CTAD选择正确的std::function
为您提供特化,无需重复类型:
compose(std::function{[] (B b, C c) -> A { return {}; }});
您还可以将此结构移至 compose
定义:
template<typename F>
void compose(F&& f) {
auto fn = std::function{std::forward<F>(f)};
// use fn here as before
};
以便调用
compose([] (B b, C c) -> A { return {}; });
足够了。
请注意,这两种情况都不适用于通用 lambda,并且还要注意,如果替换 std::function
,它们也不起作用。与您的compose_fn_t
别名,因为 CTAD 不是对别名进行的。
关于c++ - "candidate template ignored: could not match ..."用于模板函数的模板函数参数,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/59356874/
根据我的理解,INSERT IGNORE 插入一个新条目,如果它不存在,如果存在,则忽略它。所以我已经尝试这样做了一段时间,但似乎没有用。这是我的尝试: insert insert ignore in
出于某种奇怪的原因,StyleCop 不尊重我的文档规则设置。考虑以下代码: internal class SomeClass { public SomeClass() { }
我有一个带有字符串参数和 bool 返回值的方法。我想确保它总是返回 true 我试过了 myClass.Foo(A.Ignored) .WithReturnType() .Returns(tru
这个问题在这里已经有了答案: Filter invoke twice when register as Spring bean (2 个答案) 关闭 1 年前。 我们正面临 SpringSecuri
将 iconv 与 //TRANSLIT//IGNORE 一起使用从 utf8 转换为 ascii 工作正常;它根据当前语言环境(在我的情况下为 de_DE)将不可转换的字符替换为正确的音译: > e
我有这样的查询: $query="INSERT IGNORE INTO mytable (key, word1, word2) VALUES (1, 'abc', 'def')"; 如果因为主键已经存
我有一个 MySQL 表,如下所示: `id` int(10) unsigned NOT NULL AUTO_INCREMENT, `company_id` int(8) unsigned NOT N
git blame --ignore-revs-file显然是现代 Git 中存在的一个选项。 只有一个问题。它不起作用。 或者至少,它对我不起作用: 您可以将其添加到 shell 脚本中: mkdi
使用docker stack deploy,我可以看到如下信息: Ignoring unsupported options: restart 是否意味着重启政策没有到位? 是否必须在撰写文件之外指定它
我正在尝试检查被 git 忽略的文件,我发现 2 个命令显示相同的结果。 哪个是 git check-ignore * git ls-files --others --ignored --exclud
我想为我的 bash 设置和插件等创建一个 git 仓库。我忽略了所有内容(第 0 行),然后在 repo 中手动添加了我想要的文件/文件夹。 (我必须这样做,因为 repo 在我的 ~ 文件夹中。)
我们有一个集中式开发服务器,每个人都在本地结账处工作。我们如何仅忽略本地存储库中的特定目录,而不忽略集中存储库中的特定目录? 一些背景资料: 该项目是一个Drupal站点,该站点的文件目录一直在变化。
当我向 svn:ignore 条目添加一些东西时,它是存储在存储库中还是存储在我的本地副本中? (在 Tortoise 中,为什么添加到 svn:ignore 的项目会出现在我的更改列表中?) 最佳答
我想确保当我从 svn 结帐时,某些 ide 项目元数据文件没有更新。不幸的是,在创建项目时它已被 checkin 项目,但我有我自己的这些文件版本。 svn:ignore 是否也会在更新期间忽略文件
我有一个目录,其中包含具有各种扩展名的文件。我想在 svn:ignore 中执行与以下 .gitignore 代码等效的操作: *.* # ignore everything !.htaccess #
我在实时服务器上有一个大型项目,我想将其提交给 svn 存储库。我已使用 svn add 将所有文件和目录添加到存储库中. 问题是我想提交一个文件夹结构,但没有它的内容。文件夹名称是: /home/m
当我使用 svn:ignore 从 SVN 中排除文件时然后另一个人尝试提交排除的文件, svn 阻止他?或者他必须在他的电脑上执行相同的命令? 最佳答案 想象一下,如果您有一堆 Java 文件( *
如果已经有一个带有_id的文档,是否可以不重新索引文档?也许像MySQL中的INSERT IGNORE查询一样? 我使用批量API,并且重新索引需要很长时间,因此我只想索引丢失的文档。 可能吗? 最佳
我的房间有问题。 我正在使用带有 Gson 转换器的改造作为其余 api,我想与房间分享 pojos。一般来说它可以工作,但在某些情况下我需要忽略一些字段,因为我有对象列表。我尝试使用 @Ignore
我正在使用 vba 代码从数组中查找值。有些结果是“#N/A”,并且单元格的左上角会出现一个绿色三角形。我想自动删除绿色三角形(忽略错误),但“#N/A”结果应该保留,只需删除绿色三角形即可。 有人知
我是一名优秀的程序员,十分优秀!