0
  • 聊天消息
  • 系统消息
  • 评论与回复
登录后你可以
  • 下载海量资料
  • 学习在线课程
  • 观看技术视频
  • 写文章/发帖/加入社区
会员中心
创作中心

完善资料让更多小伙伴认识你,还能领取20积分哦,立即完善>

3天内不再提示

Tokio 的基本用法

科技绿洲 来源:TinyZ 作者:TinyZ 2023-09-19 16:05 次阅读

Tokio 是一个异步 I/O 框架,它提供了一种高效的方式来编写异步代码。它使用 Rust 语言的 Futures 库来管理异步任务,并使用 Reactor 模式来处理 I/O 事件。

本系列 Tokio 篇将由浅入深的从基础到实战,以一个完整的 Rust 语言子系列讲述网络编程

为什么要使用 Tokio?

在 Rust 中,使用异步编程可以提高程序的性能和响应速度,但是异步编程往往需要编写大量的样板代码和复杂的控制流程。Tokio 提供了一种简单的方式来编写异步代码,它使用 Futures 库来管理异步任务,并提供了一组工具来处理异步 I/O 事件。

如何使用 Tokio?

使用 Tokio 编写异步代码需要掌握以下几个概念:

  • • Future:表示一个异步任务,可以理解为一个异步函数的返回值;
  • • Task:表示一个异步任务的执行上下文,可以理解为一个异步函数的执行环境;
  • • Reactor:表示一个 I/O 事件的处理器,可以理解为一个事件循环;
  • • Runtime:表示一个异步任务的执行环境,可以理解为一个异步函数的运行时环境。

下面我们将使用 Tokio 编写一个最基础的服务器和客户端程序,以便了解 Tokio 的基本用法。

编写服务器

我们将编写一个简单的 PingPong 服务器,它接收客户端的 Ping 请求,并返回 Pong 响应。首先,我们需要创建一个异步任务来处理客户端的请求。我们可以使用 Tokio 提供的async关键字来定义一个异步函数:

async fn handle_client(mut stream: TcpStream) - > Result< (), Box< dyn Error >> {
    // ...
}

这个异步函数接收一个TcpStream对象,表示一个客户端连接。我们可以在函数内部处理客户端的请求,并返回一个Result对象表示异步任务的执行结果。在处理客户端请求之前,我们需要先向客户端发送一个欢迎消息:

async fn handle_client(mut stream: TcpStream) - > Result< (), Box< dyn Error >> {
    println!("new client connected");

    let mut buf = [0; 1024];
    stream.write_all(b"Welcome to the PingPong server!n").await?;

    // ...
}

在发送欢迎消息之后,我们需要不断地从客户端读取数据,并返回 Pong 响应。我们可以使用一个无限循环来实现这个功能:

async fn handle_client(mut stream: TcpStream) - > Result< (), Box< dyn Error >> {
    println!("new client connected");

    let mut buf = [0; 1024];
    stream.write_all(b"Welcome to the PingPong server!n").await?;

    loop {
        let n = stream.read(&mut buf).await?;
        if n == 0 {
            break;
        }
        stream.write_all(b"Pongn").await?;
    }

    println!("client disconnected");
    Ok(())
}

在循环中,我们使用stream.read()方法从客户端读取数据,并使用stream.write_all()方法向客户端发送 Pong 响应。如果客户端关闭了连接,我们就退出循环并返回Ok(())表示异步任务执行成功。

现在我们已经编写了一个异步任务来处理客户端请求,接下来我们需要创建一个 Reactor 来处理 I/O 事件。我们可以使用 Tokio 提供的TcpListener对象来监听客户端连接:

#[tokio::main]
async fn main() - > Result< (), Box< dyn Error >> {
    let addr = "127.0.0.1:8080";
    let listener = TcpListener::bind(addr).await?;
    println!("listening on {}", addr);

    loop {
        let (stream, _) = listener.accept().await?;
        tokio::spawn(async move {
            if let Err(e) = handle_client(stream).await {
                eprintln!("error: {}", e);
            }
        });
    }
}

main函数中,我们首先创建一个TcpListener对象来监听客户端连接。然后我们使用一个无限循环来等待客户端连接,并使用listener.accept()方法来接收客户端连接。当有新的客户端连接时,我们就创建一个新的异步任务来处理客户端请求,并使用tokio::spawn()方法将任务提交到 Reactor 中执行。

现在我们已经完成了一个最基础的 PingPong 服务器,可以使用cargo run命令来运行程序,并使用 telnet 命令来测试服务器:

$ cargo run
    Finished dev [unoptimized + debuginfo] target(s) in 0.08s
     Running `target/debug/pingpong`
listening on 127.0.0.1:8080
$ telnet localhost 8080
Trying 127.0.0.1...
Connected to localhost.
Escape character is '^]'.
Welcome to the PingPong server!
ping
Pong
ping
Pong
^]
telnet > quit
Connection closed.

编写客户端

现在我们已经编写了一个最基础的 PingPong 服务器,接下来我们将编写一个客户端程序来连接服务器并发送 Ping 请求。首先,我们需要创建一个异步任务来连接服务器:

async fn connect() - > Result< (), Box< dyn Error >> {
    let addr = "127.0.0.1:8080";
    let mut stream = TcpStream::connect(addr).await?;
    println!("connected to {}", addr);

    // ...
}

这个异步任务使用TcpStream::connect()方法来连接服务器,并返回一个Result对象表示连接结果。在连接成功之后,我们可以向服务器发送一个 Ping 请求:

async fn connect() - > Result< (), Box< dyn Error >> {
    let addr = "127.0.0.1:8080";
    let mut stream = TcpStream::connect(addr).await?;
    println!("connected to {}", addr);

    stream.write_all(b"Pingn").await?;
    let mut buf = [0; 1024];
    let n = stream.read(&mut buf).await?;
    let pong = std::str::from_utf8(&buf[..n])?;
    println!("{}", pong);

    Ok(())
}

在发送 Ping 请求之后,我们使用stream.read()方法从服务器读取响应,并使用std::str::from_utf8()方法将响应转换为字符串。最后,我们将响应打印到控制台上,并返回Ok(())表示异步任务执行成功。

现在我们已经编写了一个异步任务来连接服务器并发送 Ping 请求,接下来我们需要在main函数中启动这个任务:

#[tokio::main]
async fn main() - > Result< (), Box< dyn Error >> {
    connect().await?;
    Ok(())
}

现在我们已经完成了一个最基础的 PingPong 客户端,可以使用cargo run命令来运行程序,并查看控制台输出:

$ cargo run
    Finished dev [unoptimized + debuginfo] target(s) in 0.08s
     Running `target/debug/pingpong`
connected to 127.0.0.1:8080
Pong

完整代码

最后,我们将完整的服务器和客户端代码放在一起,以便读者参考:

use std::error::Error;
use tokio::io::{AsyncReadExt, AsyncWriteExt};
use tokio::net::{TcpListener, TcpStream};

async fn handle_client(mut stream: TcpStream) - > Result< (), Box< dyn Error >> {
    println!("new client connected");

    let mut buf = [0; 1024];
    stream.write_all(b"Welcome to the PingPong server!n").await?;

    loop {
        let n = stream.read(&mut buf).await?;
        if n == 0 {
            break;
        }
        stream.write_all(b"Pongn").await?;
    }

    println!("client disconnected");
    Ok(())
}

#[tokio::main]
async fn main() - > Result< (), Box< dyn Error >> {
    let addr = "127.0.0.1:8080";
    let listener = TcpListener::bind(addr).await?;
    println!("listening on {}", addr);

    loop {
        let (stream, _) = listener.accept().await?;
        tokio::spawn(async move {
            if let Err(e) = handle_client(stream).await {
                eprintln!("error: {}", e);
            }
        });
    }
}

async fn connect() - > Result< (), Box< dyn Error >> {
    let addr = "127.0.0.1:8080";
    let mut stream = TcpStream::connect(addr).await?;
    println!("connected to {}", addr);

    stream.write_all(b"Pingn").await?;
    let mut buf = [0; 1024];
    let n = stream.read(&mut buf).await?;
    let pong = std::str::from_utf8(&buf[..n])?;
    println!("{}", pong);

    Ok(())
}

#[tokio::main]
async fn main() - > Result< (), Box< dyn Error >> {
    connect().await?;
    Ok(())
}

总结

通过本文的介绍,我们了解了 Tokio 的基本用法,并编写了一个最基础的 PingPong 服务器和客户端程序。Tokio 提供了一种简单的方式来编写异步代码,可以帮助我们提高程序的性能和响应速度。在实际开发中,我们可以根据需要使用 Tokio 提供的各种工具来编写更加复杂的异步程序。

声明:本文内容及配图由入驻作者撰写或者入驻合作网站授权转载。文章观点仅代表作者本人,不代表电子发烧友网立场。文章及其配图仅供工程师学习之用,如有内容侵权或者其他违规问题,请联系本站处理。 举报投诉
  • 程序
    +关注

    关注

    116

    文章

    3779

    浏览量

    80896
  • 代码
    +关注

    关注

    30

    文章

    4762

    浏览量

    68408
  • 网络编程
    +关注

    关注

    0

    文章

    71

    浏览量

    10067
  • Tokio
    +关注

    关注

    0

    文章

    12

    浏览量

    52
收藏 人收藏

    评论

    相关推荐

    SQLx的基础用法和进阶用法

    SQLx是一个Rust语言的异步SQL数据库访问库,支持多种数据库,包括PostgreSQL、MySQL、SQLite等。本教程将以SQLite为例,介绍SQLx的基础用法和进阶用法。 基础用法
    的头像 发表于 09-19 14:29 2272次阅读

    SQLx在Rust语言中的基础用法和进阶用法

    SQLx是一个Rust语言的异步SQL执行库,它支持多种数据库,包括MySQL、PostgreSQL、SQLite等。本教程将以MySQL数据库为例,介绍SQLx在Rust语言中的基础用法和进阶用法
    的头像 发表于 09-19 14:32 5157次阅读

    Stream模块的基础用法和进阶用法

    在 Rust 语言中,Tokio 是一个非常流行的异步编程框架。它提供了一系列的模块,其中最常用的就是 Stream 模块。Stream 模块允许我们以异步的方式处理数据流,这在很多情况下非常
    的头像 发表于 09-19 15:33 1164次阅读

    什么是Tokio模块 Channel?

    Rust 语言是一种系统级编程语言,它具有强类型和内存安全性。Rust 语言中的 Tokio 模块是一个异步编程库,它提供了一种高效的方式来处理异步任务。其中,channel 是 Tokio 模块中
    的头像 发表于 09-19 15:57 930次阅读

    AsyncRead和AsyncWrite 模块进阶用法示例

    们。 基础用法 从文件中读取数据 use tokio::fs::File; use tokio::io::{ self , AsyncReadExt}; #[tokio:
    的头像 发表于 09-20 11:41 851次阅读

    状态机原理及用法

    状态机原理及用法状态机原理及用法状态机原理及用法
    发表于 03-15 15:25 0次下载

    使用tokio实现一个简单的Client和Server通讯模型

    本系列是关于用Rust构建一个KV Server的系列文章,内容包括用tokio做底层异步网络通讯、使用toml文件做配置、protobuf做传输协议、内存/RockDB做数据存储、事件通知、优雅关机、并发连接限制及测量监控等。
    的头像 发表于 09-09 09:45 2255次阅读

    WasmEdge增加了Tokio支持

    看:https://wasmer.io/posts/wasmer-takes-webassembly-libraries-manistream-with-wai WasmEdge增加了Tokio 支持
    的头像 发表于 12-05 11:55 821次阅读

    Tokio中hang死所有worker的方法

    原因是 tokio 里的待执行 task 不是简单的放到一个 queue 里,除了 runtime 内共享的,可被每个 worker 消费的run_queue[2],每个 worker 还有一个自己的 lifo_slot[3],只存储一个最后被放入的 task (目的是减小调度延迟)。
    的头像 发表于 02-03 16:26 963次阅读

    文盘Rust -- 用Tokio实现简易任务池

    59执行完后面就没有输出了,如果把max_task设置为2,情况会好一点,但是也没有执行完所有的异步操作,也就是说在资源不足的情况下,Tokio会抛弃某些任务,这不符合我们的预期。
    的头像 发表于 04-09 10:24 1281次阅读

    Tokio 模块的优雅停机机制

    在进行高并发、网络编程时,优雅停机是一个非常重要的问题。在 Rust 语言中,Tokio 是一个非常流行的异步编程框架,它提供了一些优雅停机的机制,本文将围绕 Tokio 模块的优雅停机进行详细
    的头像 发表于 09-19 15:26 604次阅读

    如何使用Tokio 和 Tracing模块构建异步的网络应用程序

    在 Rust 语言中,Tokio 是一个非常流行的异步运行时,它提供了高效的异步 I/O 操作和任务调度。而 Tracing 则是一个用于应用程序跟踪的框架,它可以帮助我们理解应用程序的行为和性能
    的头像 发表于 09-19 15:29 649次阅读

    基于select!宏的进阶用法

    Tokio 是一个基于 Rust 语言的异步编程框架,它提供了一组工具和库,使得异步编程变得更加容易和高效。其中最重要的组件之一就是 select!宏。 select!宏是 Tokio 中的一个核心
    的头像 发表于 09-19 15:35 619次阅读

    如何使用 Tokio 模块的Channel

    Channel 是一种在多线程环境下进行通信的机制,可以让线程之间互相发送消息和共享数据。Rust 语言中的 Tokio 模块提供了一种异步的 Channel 实现,使得我们可以在异步程序中方
    的头像 发表于 09-19 15:38 650次阅读

    tokio模块channel中的使用场景和优缺点

    Rust 语言的 tokio 模块提供了一种高效的异步编程方式,其中的 channel 模块是其核心组件之一。本教程将介绍 tokio 模块 channel 的除了上文提到的 mspc
    的头像 发表于 09-19 15:54 756次阅读