我正在使用Tokio替换使用异步等效的Rust编写的同步套接字代码。 Tokio将future用于异步活动,因此任务链接在一起并排队到执行程序上,由线程池执行。
我想要做的基本伪代码是这样的:
let tokio::net::listener = TcpListener::bind(&sock_addr).unwrap();
let server_task = listener.incoming().for_each(move |socket| {
let in_buf = vec![0u8; 8192];
// TODO this should happen continuously until an error happens
let read_task = tokio::io::read(socket, in_buf).and_then(move |(socket, in_buf, bytes_read)| {
/* ... Logic I want to happen repeatedly as bytes are read ... */
Ok(())
};
tokio::spawn(read_task);
Ok(())
}).map_err(|err| {
error!("Accept error = {:?}", err);
});
tokio::run(server_task);
这个伪代码只会执行一次我的任务。我该如何连续运行?我希望它执行然后一次又一次地执行等。我只希望它在恐慌或有错误结果代码时停止执行。最简单的方法是什么?
答案 0 :(得分:2)
使用loop_fn
应该有效:
let read_task =
futures::future::loop_fn((socket, in_buf, 0), |(socket, in_buf, bytes_read)| {
if bytes_read > 0 { /* handle bytes */ }
tokio::io::read(socket, in_buf).map(Loop::Continue)
});
答案 1 :(得分:0)
一种完成此任务而不必与类型系统战斗的干净方法是使用tokio-codec
条板箱;如果您想以字节流的形式与阅读器进行交互,而不是定义编解码器,则可以使用tokio_codec::BytesCodec
。
use tokio::codec::Decoder;
use futures::Stream;
...
let tokio::net::listener = TcpListener::bind(&sock_addr).unwrap();
let server_task = listener.incoming().for_each(move |socket| {
let (_writer, reader) = tokio_codec::BytesCodec::new().framed(socket).split();
let read_task = reader.for_each(|bytes| {
/* ... Logic I want to happen repeatedly as bytes are read ... */
});
tokio::spawn(read_task);
Ok(())
}).map_err(|err| {
error!("Accept error = {:?}", err);
});
tokio::run(server_task);