rust - 具有包含异步 block 的闭包的异步方法无法推断适当的生存期

标签 rust closures lifetime hyper

我正在尝试使用hyper创建小型HTTP路由器,但是在启动服务器时遇到了一些困难。

以下功能是我遇到问题的地方。

依存关系:

[dependencies]
hyper = "0.13"
tokio = { version = "0.2", features = ["full"] }
use std::net::SocketAddr;
use std::convert::Infallible;
use hyper::Server;
use hyper::service::{make_service_fn, service_fn};
use hyper::{Body, Request, Response};

#[tokio::main]
async fn main() {
    let router = Router::new(SocketAddr::from(([127, 0, 0, 1], 80)));
    router.start();
}

pub struct Router<'a> {
    addr: SocketAddr,
}

impl<'a> Router<'a> {
    pub fn new(addr: SocketAddr) -> Self {
        Self {
            addr,
        }
    }

    pub async fn start(&self) {
        let make_svc = make_service_fn(|_| async {
            Ok::<_, Infallible>(service_fn(|req| self.route(req)))
        });

        let server = Server::bind(&self.addr).serve(make_svc);

        println!("Listening on http://{}", self.addr);

        if let Err(e) = server.await {
            eprintln!("Server error: {}", e);
        }
    }

    pub async fn route(&self, req: Request<Body>) -> Result<Response<Body>, Infallible> {
        // TODO

        Ok(Response::new(
            "Hello, world!".into()
        ))
    }
}

我不能使用Ok::<_, Infallible>(service_fn(self.route),因为参数之一是&self。我发现的解决方案的问题在于,编译器无法推断出适当的生存期,而且我也不知道如何解决该问题。
error: cannot infer an appropriate lifetime
  --> src\main.rs:24:24
   |
24 |     pub async fn start(&self) {
   |                        ^^^^^ ...but this borrow...
...
29 |         let server = Server::bind(&self.addr).serve(make_svc);
   |                                               ----- this return type evaluates to the `'static` lifetime...
   |
note: ...can't outlive the lifetime `'_` as defined on the method body at 24:24
  --> src\main.rs:24:24
   |
24 |     pub async fn start(&self) {
   |                        ^

error: aborting due to previous error

最佳答案

您的make_svc使用&self,但是仅在start函数结束之前临时借用该变量。该服务器可能存在于start函数外部,因此它不能在其中引用借用的变量。

当对诸如服务器之类的长期使用闭包时,不能使用临时引用。请改用Arc,例如start(self: Arc<Self>),并在调用该方法之前将对象实际包装在Arc::new()中。

关于rust - 具有包含异步 block 的闭包的异步方法无法推断适当的生存期,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/61740161/

相关文章:

rust - 如何在 Dioxus 中添加全局 keydown 事件监听器)?

rust - 如何复制/克隆既不派生的结构?

php - "use"和将参数传递给 Controller ​​函数之间的区别

c# - 一段时间后是否重置静态变量

mysql - 在 mac OS 中安装 mysqlclient 库

string - 在 Vec<i8> 和 &str 之间转换

swift - 如何有一个可选的尾随闭包?

javascript - 如何从内部函数访问外部作用域?

rust - 解决通过引用获取参数的闭包的类型不匹配

rust - 指定特征边界时如何指定临时生命周期?