- 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/
我对 Clojure 和函数式编程有了大约一周的了解——我的所有背景都是 OOP。我想利用 Clojure 备受争议的易读性和固有逻辑,但现在我不知道我是否成功地做到了这一点,只是没有完全理解它,或者
场景: val col: IndexedSeq[Array[Char]] = for (i = 0 && arr(last.y)(west) == '.') { arr(last.y)(w
我正面临 AngularJS、服务和范围的“问题”。 这不是一个真正的问题(我找到了几种使其工作的方法),但我想知道我是否在做正确的事情,或者我正在做的事情是否会导致将来出现问题 我有一个保存一些全局
进行以下数据结构转换的“Rubyist”方法是什么: 我有 incoming = [ {:date => 20090501, :width => 2}, {:
如何在 go 中编写返回集合最小值的函数?我不只是在寻找解决方案(我知道我可以在遍历第一个元素时只初始化最小值,然后设置一个我初始化最小值的 bool 变量),而是一个惯用的解决方案。由于 go 没有
好的,我知道我应该对我的特定应用程序进行基准测试,等等,但是: -Xmx 的默认 JVM 设置、默认垃圾收集器等,对于大多数典型的 Java 程序来说是合理的默认设置,并且可能不适合惯用的 Scala
既然 shared_ptr 在 tr1 中,你认为 std::auto_ptr 的使用会发生什么?它们都有不同的用例,但 auto_ptr 的所有用例也都可以用 shared_ptr 解决。你会放弃
这个问题在这里已经有了答案: What are the differences between type() and isinstance()? (8 个回答) 关闭 9 年前。 我需要知道 Pyth
在指定和创建数字函数时,是否有关于何时返回 null 以及何时返回 NaN 的任何 C# 惯用准则,当两者似乎都是有效输出时。 导致这个问题的具体例子是我正在为 Enumerable 集合创建一个百分
这个问题在这里已经有了答案: Retrieving the top 100 numbers from one hundred million of numbers [duplicate] (12 个
我可以通过反射检索方法,以某种方式将其与目标对象结合起来,并将其作为看起来像 Scala 中的函数的东西返回(即您可以使用括号调用它)吗?参数列表是可变的。它不一定是“一流”函数(我已经更新了问题),
我是一名优秀的程序员,十分优秀!