gpt4 book ai didi

rust - 将非复制变量移动到异步闭包 : captured variable cannot escape `FnMut` closure body

转载 作者:行者123 更新时间:2023-12-04 12:09:26 27 4
gpt4 key购买 nike

我正在尝试获取 clokwerk安排异步函数每 X 秒运行一次。
The docs显示这个例子:

// Create a new scheduler
let mut scheduler = AsyncScheduler::new();
// Add some tasks to it
scheduler
.every(10.minutes())
.plus(30.seconds())
.run(|| async { println!("Simplest is just using an async block"); });
// Spawn a task to run it forever
tokio::spawn(async move {
loop {
scheduler.run_pending().await;
tokio::time::sleep(Duration::from_millis(100)).await;
}
});
我的初步尝试:
    let config2 = // define a Config struct, Config
let pg_pool2 = // get a sqlx connection pool, Pool<Postgres>

//I assume I need shared references so I use Arc
let pg_pool2 = Arc::new(pg_pool2);
let config2 = Arc::new(config2);

let mut scheduler = AsyncScheduler::new();

scheduler.every(5.seconds()).run(|| async {
println!("working!");
pull_from_main(pg_pool2.clone(), config2.clone()).await;
});

tokio::spawn(async move {
loop {
scheduler.run_pending().await;
tokio::time::sleep(Duration::from_millis(100)).await;
}
});
编译器提示 pg_pool2config2可能会超过借入的值(value),并建议添加 move .公平的。让我们试试看。
我的第二次尝试:
    //rest the same
scheduler.every(5.seconds()).run(move || async {
//rest the same
这次我得到了一个我无法自己破译的错误:
error: captured variable cannot escape `FnMut` closure body
--> src/main.rs:80:46
|
75 | let pg_pool2 = Arc::new(pg_pool2);
| -------- variable defined here
...
80 | scheduler.every(5.seconds()).run(move || async {
| ____________________________________________-_^
| | |
| | inferred to be a `FnMut` closure
81 | | println!("working!");
82 | | pull_from_main(pg_pool2.clone(), config2.clone()).await;
| | -------- variable captured here
83 | | });
| |_____^ returns an `async` block that contains a reference to a captured variable, which then escapes the closure body
|
= note: `FnMut` closures only have access to their captured variables while they are executing...
= note: ...therefore, they cannot allow references to captured variables to escape
有人可以帮助我了解出了什么问题以及如何解决吗?
注意:我之前看过这个问题,但我正在努力将答案应用于我的案例。
  • here涉及一个可变变量,我没有。
  • here解决方案是将变量作为 fn 参数包含在内,我不能这样做,因为闭包不需要参数。当我不这样做时,它似乎也从闭包返回一个值,这会导致问题。

  • 我也是初学者,所以也许他们确实申请了,但我看不到联系:)

    最佳答案

    为了理解发生了什么,我将重新格式化代码,使其更加清晰和明确:
    您的原始代码:

      scheduler
    .every(5.seconds())
    .run(move || async {
    do_something(arc.clone());
    });
    相当于:
      scheduler
    .every(5.seconds())
    .run(move || {
    return async {
    do_something(arc.clone());
    }
    });
    所以你创建了一个闭包,它的类型是 FnMut (并返回一个实现 Future 的类型)。这意味着您的闭包可以被多次调用,并且每次调用都应该产生一个新的 future 。但是 return async{} 移动 您的 Arc出闭包,这意味着它只能被调用一次。想象一下,你的钱包里有一张 10 美元的钞票。如果你把它拿出来花掉,那你就不能再拿出来花掉了,因为它根本就不存在了。
    那么我们该如何解决呢?这实际上很容易 - 你必须克隆你的 Arc在将其移至 async 之前堵塞。因此,您将只移动克隆:
        let arc = Arc::new(whatever);   
    scheduler
    .every(5.seconds())
    .run(move || {
    // Clone the arc and move the clone!!!
    // The original arc will remain in the closure,
    // so it can be called multiple times.
    let x = arc.clone();
    async move {
    do_something(x);
    }
    });

    关于rust - 将非复制变量移动到异步闭包 : captured variable cannot escape `FnMut` closure body,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/67960931/

    27 4 0
    Copyright 2021 - 2024 cfsdn All Rights Reserved 蜀ICP备2022000587号
    广告合作:1813099741@qq.com 6ren.com