- html - 出于某种原因,IE8 对我的 Sass 文件中继承的 html5 CSS 不友好?
- JMeter 在响应断言中使用 span 标签的问题
- html - 在 :hover and :active? 上具有不同效果的 CSS 动画
- html - 相对于居中的 html 内容固定的 CSS 重复背景?
Rust 中函数接受闭包作为参数或返回闭包的惯用方式是什么?
我认为至少可以通过以下三种方式来完成:
// 1
pub fn run_with_envs_guard1(envs: &HashMap<&str, &str>, f: &dyn FnOnce()) {}
// 2
pub fn run_with_envs_guard2(envs: &HashMap<&str, &str>, f: Box<dyn FnOnce()>) {}
// 3
pub fn run_with_envs_guard3<F: FnOnce()>(envs: &HashMap<&str, &str>, f: F) {}
这三种方式真的有区别吗?如果是,请帮助澄清,我应该选择哪种方式更惯用?
我仍在学习 rust,很抱歉,如果以上所有方法都是一些不好/奇怪的事情。
也许是一个更具体的问题,为什么在1
和2
中我需要dyn
关键字,但在3
中code> 我不知道,据我了解,这些都需要动态调度,是吗?因为实际功能无法在编译时确定
最佳答案
阿卜杜勒回答了你问题的前半部分(我完全同意他所说的),所以我将尝试后半部分。
如果您想从函数返回闭包,则无法返回类型参数,因为这意味着您将返回 any FnOnce
的实例。 ,由调用者选择。您无法返回 &FnOnce
,因为您(通常)需要将所有权传递给调用者。你可以让它与Box<FnOnce>
一起工作,但这往往使用起来很笨拙。当从函数返回闭包时,我偏向 impl
trait syntax .
pub fn test() -> impl FnOnce() {
|| { println!("It worked!") }
}
在参数位置,写 impl FnOnce()
因为某物的类型相当于定义类型参数,正如阿卜杜勒在他的回答中所做的那样。然而,在返回位置,这是一个全新的功能,它返回一个不透明的值。它说“我正在返回一个 FnOnce
,但我不会告诉你它是哪一个”。它与特征对象的概念相同,但没有将其扔进盒子的开销。
回复您的编辑
i don't, from my understanding, these all need dynamic dispatching, is it? as the actual function cannot be determined in compiling time
这实际上不一定是真的。如果您看到dyn
关键字,那么肯定会发生动态(运行时)调度。不过,为了理解您的另一个示例,让我们考虑一个没有 FnOnce
包袱的简单特征。 .
pub trait MyTrait {}
struct Foo;
struct Bar;
impl MyTrait for Foo {}
impl MyTrait for Bar {}
pub fn example<T: MyTrait>(_arg: T) {
println!("It works!");
}
fn main() {
example(Foo);
example(Bar);
}
我声称这里没有发生动态调度。 Rust 使用类型参数对函数进行单态化。这意味着example
就像C++中的模板函数。它的每个实例化最终都会成为一个单独的函数。所以,实际上,在 Rust 编译期间,这最终会更像
struct Foo;
struct Bar;
pub fn example1(_arg: Foo) {
println!("It works!");
}
pub fn example2(_arg: Foo) {
println!("It works!");
}
fn main() {
example1(Foo);
example2(Bar);
}
两个不相关的函数碰巧做了类似的事情。 Rust 静态地解析所有链接,因此运行时不会发生调度。事实上,我们可以证明这一点。获取我刚刚在上面发布的代码,并使用调试符号( rustc -g filename.rs
)对其进行编译。然后使用像 nm
这样的工具(默认情况下在大多数 Linux 机器上可用)列出链接器表中的所有符号。假设您没有打开任何优化,您应该看到两个 example
功能。这就是它们在我的链接器中的样子
0000000000005340 t _ZN10code7example17h46383f9ad372dc94E
00000000000053a0 t _ZN10code7example17h97b400359a146fcaE
或者,与 nm -C
解析函数名称
0000000000005340 t code::example
00000000000053a0 t code::example
两个不同的函数,每个函数都采用特定类型的具体参数。
您的建议FnOnce
会以同样的方式工作。
pub fn run_with_envs_guard3<F: FnOnce()>(envs: &HashMap<&str, &str>, f: F) {}
Rust 中的每个闭包都有不同的类型,因此每次调用此函数时,都会生成 run_with_envs_guard3
的新版本将被制作,专门用于该关闭。该新函数将确切地知道如何处理您刚刚给出的闭包。在 99% 的情况下,如果您打开了优化,这些虚构的本地函数将被内联并优化掉,因此不会造成任何损害。但这里没有动态调度。
在另外两个示例中,我们有 dyn FnOnce
,这更像是您所期望的传统面向对象语言。 dyn FnOnce
包含一个指向某个函数的动态指针,该函数将按照您期望的方式在运行时调度。
关于rust - rust 中函数接受闭包作为参数或返回闭包的惯用方式是什么?,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/69956268/
我正在本地编程应用程序,但是当我迁移到服务器时,使用此行出现解析错误: if(!is_array($data[array_keys($data)[0]])) 返回值: Parse error: syn
关闭。这个问题是opinion-based 。目前不接受答案。 想要改进这个问题吗?更新问题,以便 editing this post 可以用事实和引文来回答它。 . 已关闭 9 年前。 Improv
我已经开始尝试用 C++ 学习 Winsock,但我遇到了一些问题。我首先遵循 MSDN 上的 Winsock 示例(一个基本的客户端-服务器)。 http://msdn.microsoft.com/
我有一段使用 epoll 的代码,但它有问题。当我运行它时,它给出输出:服务器套接字()没问题......服务器绑定(bind)()没问题......3个4个接受:无效参数 我在 ubuntu lin
我正在寻找一种方法来接受 $_GET 变量作为文件路径,例如 - /page/test将转换为 page.php?page=test .我已经搜索过,但找不到任何可以帮助我的东西,因此我在这里问。 我
我想要一个只接受从 0 到 9 和减号的浮点数的正则表达式。 请帮忙。 最佳答案 ^[-+]?[0-9]*\.?[0-9]+$ ^ - 字符串开头 [-+]? - 0 或 1 符号指示符 [0-9]*
请问如何接受\r\n无需将其更改为 \\r\\n , 与 fgets . 我想让程序翻译 \r\n到换行符而不是将其打印为字符串。 当前代码: char buff[1024]; printf("Msg
我正在编写一个 Cocoa 应用程序,该应用程序需要在其 Dock 图标上接受已安装卷的滴落。它不是基于文档的;我打算将每个卷分派(dispatch)到 application:openFiles 中
我在 SQLite 中发现了这种意外行为。 SQLite 似乎接受 SQL 连接语法中的任意关键字。如果我不小心键入了 natural join 而不是 natural join,则会生成笛卡尔积。这
我在 Windows 窗体的同一个窗体上有一个 TreeView 和一个多行文本框。我有拖放设置,以便我可以将节点从 TreeView 拖到文本框并将文本插入文本框(这是有效的)。 我想增强这一点,以
我正在创建一棵类似于 D3 Layout Tree 的树并尝试绑定(bind)我的自定义 JSON 对象。切换树节点的代码如下。 function toggleAll(d) { if (d.c
所以,我希望能够向我的 DOB 字段发送空选项。 这是我的表单生成器: ->add('birthDate', DateType::class, array( 'widg
错误可以在这里看到:http://djaffry.selfip.com:8080/ 我希望索引页接受参数,无论是 mysite.com/search/param_here 或 mysite.com/?
我想知道标准 Scala 解析器组合器是否包含一个解析器,该解析器接受 Scala 语言本身也接受的相同标识符(如 Scala 语言规范第 1.1 节中所指定)。 StdTokenParsers 特征
我暂时使用以下行在我的 Android 手机上创建 ServerSocket: socketl = new ServerSocket(port, 0, InetAddress.getByName("1
今天早上刚刚发布了一个应用程序,我在所有可能的设备和模拟器上测试了它,但是当我从商店下载它时,应用程序在启动时崩溃。我在很多设备和iOS版本上测试过,但结果都是一样的: Incident Identi
我想要 5 个名字中最长的一个。我想我应该使用 compareTo() 方法或 length()? 输出必须是这样的: enter 5 names : Joey Mark Catherine Zach
抱歉,我不熟悉泛型,我可以创建一个如下所示的泛型类: public class InfoField { } 上面的类可以接受如下值: , User> 但是我可以有接受 map 对象作为值的类吗?
我想让一个单元格等于它上面的单元格。 当我写作时 =address(row()-1;column()) 它只是写入上面单元格的名称(例如,对于 B2,它写入 $B$1) 是否有一个函数可以输入地址并放
我正在使用Asm访问java字节码来获取方法信息。看来ClassReader类的accept方法是异步调用的,所以不可能立即获取方法信息。我该怎么办? 最佳答案 实际上,accept 方法是从您自己的
我是一名优秀的程序员,十分优秀!