期望一个实现`Fn`特征的闭包,但是这个闭包只实现了`FnOnce`

时间:2018-02-13 10:41:39

标签: rust hyper

我想使用Hyper来实现Web服务。我从the hello world example复制了代码并且成功了。当我尝试向HelloWorld结构添加数据访问对象时,出现错误,我不知道如何修复它。如何将特征成员添加到Hyper服务器?

extern crate futures;
extern crate hyper;

use futures::future::Future;
use hyper::header::ContentLength;
use hyper::server::{Http, Request, Response, Service};

trait Dao {}

struct MysqlDao;

impl Dao for MysqlDao {}

struct HelloWorld {
    dao: Box<Dao>,
}

const PHRASE: &'static str = "Hello, World!";

impl Service for HelloWorld {
    // boilerplate hooking up hyper's server types
    type Request = Request;
    type Response = Response;
    type Error = hyper::Error;
    // The future representing the eventual Response your call will
    // resolve to. This can change to whatever Future you need.
    type Future = Box<Future<Item = Self::Response, Error = Self::Error>>;

    fn call(&self, _req: Request) -> Self::Future {
        // We're currently ignoring the Request
        // And returning an 'ok' Future, which means it's ready
        // immediately, and build a Response with the 'PHRASE' body.
        Box::new(futures::future::ok(
            Response::new()
                .with_header(ContentLength(PHRASE.len() as u64))
                .with_body(PHRASE),
        ))
    }
}

fn main() {
    let addr = "127.0.0.1:3000".parse().unwrap();
    let dao = Box::new(MysqlDao);
    let server = Http::new().bind(&addr, || Ok(HelloWorld { dao })).unwrap();
    server.run().unwrap();
}

错误信息:

error[E0525]: expected a closure that implements the `Fn` trait, but this closure only implements `FnOnce`
  --> src/main.rs:44:42
   |
44 |     let server = Http::new().bind(&addr, || Ok(HelloWorld { dao })).unwrap();
   |                              ----        ^^^^^^^^^^^^^^^^^^^^^^^^^
   |                              |
   |                              the requirement to implement `Fn` derives from here
   |
note: closure is `FnOnce` because it moves the variable `dao` out of its environment
  --> src/main.rs:44:61
   |
44 |     let server = Http::new().bind(&addr, || Ok(HelloWorld { dao })).unwrap();
   |                                                             ^^^

1 个答案:

答案 0 :(得分:2)

我对struct HelloWorld<'a> { dao: &'a Dao, } 结构进行了这些更改:

let server

我还将let server = Http::new() .bind(&addr, move || Ok(HelloWorld { dao: &dao })) .unwrap(); 语句更改为:

extern crate futures;
extern crate hyper;

use futures::future::Future;
use hyper::header::ContentLength;
use hyper::server::{Http, Request, Response, Service};

trait Dao {}

struct MysqlDao;

impl Dao for MysqlDao {}

struct HelloWorld<'a> {
    dao: &'a Dao,
}

const PHRASE: &'static str = "Hello, World!";

impl<'a> Service for HelloWorld<'a> {
    // boilerplate hooking up hyper's server types
    type Request = Request;
    type Response = Response;
    type Error = hyper::Error;
    // The future representing the eventual Response your call will
    // resolve to. This can change to whatever Future you need.
    type Future = Box<Future<Item = Self::Response, Error = Self::Error>>;

    fn call(&self, _req: Request) -> Self::Future {
        // We're currently ignoring the Request
        // And returning an 'ok' Future, which means it's ready
        // immediately, and build a Response with the 'PHRASE' body.
        Box::new(futures::future::ok(
            Response::new()
                .with_header(ContentLength(PHRASE.len() as u64))
                .with_body(PHRASE),
        ))
    }
}

fn main() {
    let addr = "127.0.0.1:3000".parse().unwrap();
    let dao = MysqlDao;
    let server = Http::new()
        .bind(&addr, move || Ok(HelloWorld { dao: &dao }))
        .unwrap();
    server.run().unwrap();
}

整个代码:

{{1}}