- html - 出于某种原因,IE8 对我的 Sass 文件中继承的 html5 CSS 不友好?
- JMeter 在响应断言中使用 span 标签的问题
- html - 在 :hover and :active? 上具有不同效果的 CSS 动画
- html - 相对于居中的 html 内容固定的 CSS 重复背景?
将 lambda 函数作为参数传递给仅使用(但不存储或转发)lambda 函数的函数的最佳(即最高性能、最通用)方法是什么?
选项 1:通过 const-reference 传递它是可行的方法吗?
template <typename F>
void just_invoke_func(const F& func) {
int i = 1;
func(i);
}
选项 2:还是将其作为转发引用(通用引用)传递更好?
template <typename F>
void just_invoke_func(F&& func) {
int i = 1;
func(i);
}
后者比前者有什么优势吗?
其中一种解决方案有任何危害吗?
编辑 #1:
选项 3: 正如 Yakk - Adam Nevraumont 所指出的, mutable
lambdas 不适用于选项 1。那么,另一个采用非 const 左值引用的版本呢:
template <typename F>
void just_invoke_func(F& func) {
int i = 1;
func(i);
}
问题是:与选项 3 相比,选项 2 是否有任何优势?
编辑#2:
选项 4:另一个按值获取 lambda 的版本 has been proposed .
template <typename F>
void just_invoke_func(F func) {
int i = 1;
func(i);
}
最佳答案
带有 mutable
关键字的 Lambda 有一个非常量 operator()
,并且在第一种情况下编译失败。
我认为没有理由阻止可变的 lambda。
右值 lambda 无法在非 const 左值引用情况下编译。
老实说,考虑按值获取 lambda。如果调用者想要非本地状态,他们可以通过引用捕获,或者甚至传入 std::ref
/std::cref
包装的 lambdas(如果你正在做递归调用尾调用不可优化,右值引用可能是明智的)。
template<class F>
void do_something1( F&& f );
template<class F>
void do_something2( F f );
可以通过调用 do_something2(std::cref(lambda))
(如果不可变)和 do_something2 来模拟对
(如果可变)。do_something1(lambda)
的调用(std::ref(lambda))
只有当 lambda 既是可变的又是右值时,这才行不通; 9/10 次是错误(状态被处置的可变 lambda 是不确定的),剩下的 1/10 次你可以解决它。
另一方面,您无法用 do_something1
轻松模仿 do_something2
。
do_something2
的优点是,编译器在编写 do_something2
的主体时,可以准确地知道所有状态的位置lambda 是。
[x = 0](int& y)mutable{y=++x;}
如果此 lambda 是按值传递的,则编译器在本地知道谁有权访问 x
。如果这个 lambda 是通过引用传递的,编译器必须理解调用上下文和它调用的所有代码,以便知道其他人是否修改了任何两个之间的 x
自己的代码中的表达式。
编译器通常可以比外部引用更好地优化值。
至于拷贝的成本,lambda 的移动通常并不昂贵。如果它们是 [&]
-capture,它们只是一堆状态引用。如果它们是值(value)捕获的,那么它们的状态很少是巨大且难以移动的。
在这些情况下,std::ref
或 std::cref
无论如何都会消除该成本。
(std::ref
的 operator()
传递给 lambda,因此函数永远不需要知道它已传递给 std::ref (lambda)
).
关于C++ 最佳实践 : Pass use-only (not stored) lambda argument to function by const-reference or by forwarding-reference (a. k.a.通用引用),我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/65562986/
以下操作是否完全相同: request.setAttribute(name,"visited"); requestDispatcher.forward(request,response) 和
有两个 header 用于处理请求转发。首先是 X-Forwarded-For第二个是 Forwarded . 我应该使用哪一个? X-Forwarded-For 是事实上的,而 Forwarded
看the documentation for std::forward , template constexpr T&& forward( typename std::remove_reference
给定一个手电筒的nn.Module带有预前钩,例如 import torch import torch.nn as nn class NeoEmbeddings(nn.Embedding):
为了在我的 ASP.NET 应用程序中获取客户端 IP 地址,我使用了 X-Forwarded-For,并从列表中获取第一个 IP 地址(根据我找到的信息,有一个客户端,代理1、代理2..)。但我最近
以下 header 有什么区别? X 转发主机 X 转发服务器 在 Apache 作为代理、Tomcat 作为源服务器的设置中,我在到达 Tomcat 的 HTTP 请求中看到它们。值是(这是我自己的
假设我想要一个通用的高阶函数。最常见的方法是: 一个。通用 lambda auto F1 = [](auto f) { /*...*/ } B.函数模板 template auto F2(F&& f)
为了在我的 ASP.NET 应用程序中获取客户端 IP 地址,我使用了 X-Forwarded-For,并从列表中获取第一个 IP 地址(根据我找到的信息,有一个客户端,代理 1、代理 2 ..)。但
我有一个场景,我想使用 Context.Forward 将上下文转发到另一个对话框,它工作得很好。现在,如果我更新代码以获取对话框堆栈并使用 DialogStack.Forward,则会导致 Stac
我遇到了 std::forward 的两种变体与可变参数模板参数一起使用。 template void foo(Args&&... arga) { bar(std::forward(args
考虑 std::apply 的可能实现: namespace detail { template constexpr decltype(auto) apply_impl(F &&f, Tuple &
我在 Foo Controller 中有一个需要参数的 Action 方法: public function fooAction($one, $two) { $a = one; $b
这些功能是否等效? template void foo(T && t) { bar(std::forward(t)); } template void foo2(T && t) {
在建立从 Fedora 23 到 Centos 7 的 ssh 连接时,我遇到了一个不幸的问题。命令 ssh -X user@centos7和 ssh -Y user@centos7都打印 警告:没有
作为从 Struts 1.x 迁移到 Spring Boot 的一部分,我一直在努力正确转发应有的信息。 最初的 Struts 版本依赖于 web.xml,尽管我想将该项目转换为更新的、完全基于 Ja
我使用了net = cv2.dnn.readNetFromCaffe(protoFile, WeightsFile),然后循环播放实时视频帧以使用net.forward()") bre
我注册了域名 www.a.com 然后我将屏蔽(使用 GoDaddy)流量从 www.a.com 转发到 www.really.long.url.com,因此用户在访问我的网站时只会在地址栏中看到 w
我有一个域,例如 domain.org以及地址为 http://server/node/myname 的服务器上的托管空间. 我如何映射 domain.org至 http://server/node/
我有jsp,具体取决于我正在转发到另一个页面的某些条件。 我有/myjsp/first.jsp 其代码如下, 在这段代码中,if flow come to else 条件控
前向声明和前向引用有什么区别? 在我看来,前向声明是当你声明一个尚未实现的函数时,但这不正确吗?您是否必须查看特定情况以声明案例“前向引用”或“前向声明”? 最佳答案 一个 前向声明是在实现和使用方法
我是一名优秀的程序员,十分优秀!