c++ - 是否可以将带有捕获和参数的 lambda 传递给另一个函数?如果是这样,怎么做?
问题描述
我正在尝试创建一个通用输入验证函数(在 C++17 中),它可以接收一个返回布尔值的函数指针,以测试输入是否有效。例如,如果我希望用户输入小于 6 的值,则 lambda 的主体将是return input < 6
.
问题出在我正在尝试做的事情上。我正在制作一个简单的控制台游戏来更熟悉 C++。我有一个名为 Change_Weapon 的函数,它接收用户拥有的所有武器的向量。由此产生以下 lambda:
// Variables renamed for clarity
auto validation_func = [&owned_weapon_vec](unsigned input_to_test) -> bool
{
return input_to_test > 0 && input_to_test <= owned_weapon_vec.size();
};
// Function call to input validation function
auto weapNum = valid_input<unsigned>(validation_func);
这就是我的输入验证功能的样子。
template<typename Ty>
Ty valid_input(std::function<bool(Ty)>(valid_input_function))
{
Ty input_val;
bool first_loop = true;
do
{ // Input value validiation loop
if (first_loop) first_loop = false;
else
{
std::cout << "Invalid entry value. Please re-enter: ";
std::cin.clear();
std::cin.ignore(/* Max possible value of long long */, '\n');
}
while (!(std::cin >> input_val))
{ // Input type validation loop
std::cout << "Invalid entry type. Please re-enter: ";
std::cin.clear();
std::cin.ignore(/* Max val of long long*/, '\n');
}
} while (!valid_input_function(input_val))
}
需要注意的一些事项:
- 被传递的函数将始终将 input_val 作为参数,这将是它的唯一参数
- 被传递的函数总是返回一个布尔值
- 传递的函数应该能够是匿名 lambda(或至少是命名的 lambda)
我尝试了其他各种帖子,但似乎找不到一个既涉及 lambda 捕获又涉及参数的答案。目前,Resharper 告诉我:
No viable function
Argument types: '<lambda>'.
Candidates considered:
unsigned valid_input<unsigned>(std::function<bool(unsigned)> valid_input_function)
conversion of 1st argument 'validation_func' is ill-formed: cannot convert lvalue of
type '<lambda>' to parameter type 'std::function::<bool(unsigned)>'
解决方案
你如何调用你的valid_input
函数?对我来说,它可以像这样工作:
template<typename T>
T valid_input(std::function<bool(T)> predicate)
{
T some_input(69); // I do not know where do you want to take it from
std::cout << "predicate returns: " << predicate(some_input) << std::endl;
return some_input;
}
int main()
{
size_t some_threshold(0);
size_t result = valid_input<size_t>([some_threshold](size_t input) -> bool
{
std::cout << "predicate invoked with " << input << std::endl;
return input > some_threshold;
});
std::cout << result << std::endl;
}
或者,如果您不想显式使用模板参数,valid_input
则可以将 lambda 显式分配给std::function
first:
std::function<bool(size_t)> f = [some_threshold](size_t input)
{
std::cout << "predicate invoked with " << input << std::endl;
return input > some_threshold;
};
size_t result = valid_input(f);
推荐阅读
- laravel - 使用 laravel new blog 时,'RM' 不被识别为内部或外部命令
- javascript - 仅在按下 shiftKey 时允许鼠标滚轮事件
- asp.net-core-3.1 - ASP.NET Core 3.1 读取请求正文空字符串
- kdb - 关于xkey实现
- c++ - 有什么方法可以减少 Radius 异常值移除 [pcl 异常值移除] 的执行时间?
- python - 如何缩短所有这些复选框的代码
- docusignapi - 区分同一开发者帐户中用于测试和实时环境的 webhook
- react-native - 如何在本机反应中删除材料底部选项卡的背景颜色
- typescript - 如果通过括号表示法访问函数的属性,如何让 TypeScript 出错?
- sql - 如何在包含星号的文本单元格上运行 Google 表格查询