rust - 如何在不使用 tokio_proto crate 的情况下从 tokio TCP 连接读取数据?

标签 rust rust-tokio

我正在尝试编写一个 TCP 客户端来打印传入的消息。我想出了以下代码:

extern crate bytes;
extern crate futures;
extern crate tokio_core;
extern crate tokio_io;

use futures::Future;
use tokio_core::net::TcpStream;
use tokio_core::reactor::Core;
use tokio_io::AsyncRead;
use bytes::BytesMut;

fn main() {
    let mut core = Core::new().unwrap();
    let handle = core.handle();

    let connection = TcpStream::connect(&"127.0.0.1:8081".parse().unwrap(), &handle);

    let server = connection.and_then(move |mut stream| {
        let mut buf = BytesMut::with_capacity(1000);
        stream
            .read_buf(&mut buf)
            .map(|buf| print!("Buffer {:?}", buf))
            .map_err(|e| eprintln!("Error: {}", e));
        Ok(())
    });

    core.run(server).unwrap();
}

它编译但失败并出现 Buffer NotReady 错误。

最佳答案

Rust 是一种编译型语言,这意味着您应该注意编译器生成的警告:

warning: unused `std::result::Result` which must be used
  --> src/main.rs:20:9
   |
20 | /         stream
21 | |             .read_buf(&mut buf)
22 | |             .map(|buf| print!("Buffer {:?}", buf))
23 | |             .map_err(|e| eprintln!("Error: {}", e));
   | |____________________________________________________^
   |
   = note: #[warn(unused_must_use)] on by default

此外,tokio has an entire chapter dedicated to low-level IO我假设您已经阅读过这些内容,以免您对已经知道的细节感到厌烦。

首先我们获取connection Future 并将其转换为Stream。一个流可以产生多个值——在这种情况下,我们为每次成功读取返回一个值。我们创建了 AsWeGetIt 来实现最简单的实现。

然后我们使用 Stream::for_each 打印出流的每个值。方便的是,这会执行相应的转换回 Future,这是 and_then 所需要的。

extern crate bytes;
extern crate futures;
extern crate tokio_core;
extern crate tokio_io;

use futures::{Future, Poll, Stream};
use tokio_core::net::TcpStream;
use tokio_core::reactor::Core;
use tokio_io::AsyncRead;
use bytes::BytesMut;

struct AsWeGetIt<R>(R);

impl<R> Stream for AsWeGetIt<R>
where
    R: AsyncRead,
{
    type Item = BytesMut;
    type Error = std::io::Error;

    fn poll(&mut self) -> Poll<Option<Self::Item>, Self::Error> {
        let mut buf = BytesMut::with_capacity(1000);

        self.0
            .read_buf(&mut buf)
            .map(|async| async.map(|_| Some(buf)))
    }
}

fn main() {
    let mut core = Core::new().unwrap();
    let handle = core.handle();

    let address = "127.0.0.1:8081".parse().expect("Unable to parse address");
    let connection = TcpStream::connect(&address, &handle);

    let client = connection
        .and_then(|tcp_stream| {
            AsWeGetIt(tcp_stream).for_each(|buf| {
                println!("Buffer {:?}", buf);
                Ok(())
            })
        })
        .map_err(|e| eprintln!("Error: {}", e));

    core.run(client).expect("Unable to run the event loop");
}

关于rust - 如何在不使用 tokio_proto crate 的情况下从 tokio TCP 连接读取数据?,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/46836933/

相关文章:

mongodb - 使用 mongodb-1.2.2 和 Rocket-0.5.0-rc.1 时如何解决异步不兼容问题?

rust - 在异步Rust(Tokio)中包装阻止mpsc

rust - 如何在stuctopt定义中访问运行时定义的变量?

rust - % 运算符在闭包内出错

rust - 局部变量的Rust异步问题

rust - Tokio 错误: "there is no reactor running"即使安装了 #[tokio::main] 和单个版本的 tokio

rust - 如何将参数化枚举从泛型类型映射到另一个类型?

generics - 要求泛型类型实现复制或克隆|来自泛型固定数组的泛型向量

rust - 匹配来自外部库的依赖于版本的枚举值

rust - 即使我正在发送换行符,TCP 回显服务器也从不回复