如何使用Tokio TcpListener处理WouldBlock错误?

时间:2019-07-06 17:12:32

标签: rust rust-tokio

考虑以下代码:

extern crate tokio; // Executor runtime

use tokio::prelude::*;
use tokio::net::TcpListener;
use std::net::SocketAddr;

fn main() {
    let addr = "127.0.0.1:8118".parse::<SocketAddr>().unwrap();
    let listener = TcpListener::bind(&addr)
        .expect("unable to bind TCP listener");
    tokio::run(listener.incoming()
        .map_err(|e| eprintln!("failed to accept socket; error = {:?}", e))
        .for_each(|mut socket| {
            let mut buf = vec![];
            socket.read_to_end(&mut buf).unwrap();
            println!("Received: {:#?}", buf);
            Ok(())
        })
    );
}

运行此命令并将其发送到端口8118时,出现以下错误:

thread 'tokio-runtime-worker-0' panicked at 'called `Result::unwrap()` on an `Err` value: Kind(WouldBlock)', src/libcore/result.rs:997:5

我想象有某种方法可以将套接字置于阻塞模式,或者捕获错误并对其进行处理。我想知道解决这个问题的标准,规范方法是什么。

我宁愿不阻止,因为我希望服务器在等待客户端时做其他事情,所以异步/线程解决方案将是很棒的。

1 个答案:

答案 0 :(得分:1)

您正在使用Tokio,该库的全部目的是启用异步IO。您从不不想在异步事件循环中执行阻止操作。

相反,要么全力以赴,要么完全避免异步,并使用更简单,更粗糙的线程。

Tokio的io::read_to_end创造了一个能够从套接字读取所有数据的未来:

use std::net::SocketAddr;
use tokio::{net::TcpListener, prelude::*}; // 0.1.22

fn main() {
    let addr = "127.0.0.1:8118".parse::<SocketAddr>().unwrap();
    let listener = TcpListener::bind(&addr).expect("unable to bind TCP listener");

    tokio::run(
        listener
            .incoming()
            .and_then(|s| tokio::io::read_to_end(s, vec![]))
            .map_err(|e| panic!("failed: {:?}", e))
            .for_each(|(_socket, buf)| {
                println!("Received: {:#?}", buf);
                Ok(())
            }),
    );
}

另请参阅: