百度360必应搜狗淘宝本站头条
当前位置:网站首页 > 技术文章 > 正文

使用 Rust 和 Tokio 构建 TCP 服务器示例

itomcoil 2025-05-14 14:05 54 浏览

Tokio 是一个用于构建异步应用程序的 Rust 框架,其核心特点是使用了基于事件循环的模型,可以实现高效的异步 IO 操作。在本文中,我们将通过一个示例程序来了解如何使用 Tokio 来构建一个基本的 TCP 服务器。

引入完整的 Tokio 功能集,将 tokio 的版本声明为 1.26.0 并指定 features = ["full"],如下所示:

[dependencies]
tokio = { version = "1.26.0", features = ["full"] }

这将确保所有 Tokio 的功能都可用,包括网络和 I/O 相关的模块。接下来我们看一下示例代码:

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

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

    loop {
        let (mut socket, _) = listener.accept().await?;
        tokio::spawn(async move {
            let mut buffer = [0; 1024];

            loop {
                let n = match socket.read(&mut buffer).await {
                    Ok(n) if n == 0 => break,
                    Ok(n) => n,
                    Err(e) => {
                        eprintln!("failed to read from socket; err = {:?}", e);
                        break;
                    }
                };

                if let Err(e) = socket.write_all(&buffer[0..n]).await {
                    eprintln!("failed to write to socket; err = {:?}", e);
                    break;
                }
            }

            println!("Connection closed");
        });
    }
    Ok(())
}

这段代码是一个使用Tokio实现的TCP服务器,它会监听在本地的8080端口,并接受来自客户端的连接请求。一旦连接建立,服务器将为每个客户端连接生成一个新的任务,并在任务中处理客户端发送的数据。

首先,我们使用 tokio::net::TcpListener 绑定本地地址和端口,开始监听传入连接。然后,我们使用 loop 循环来等待客户端连接,每当有新的连接进入时,使用 listener.accept().await 方法接收连接,并返回一个包含新建立连接的 TcpStream 对象和远程连接的地址的元组。

接下来,我们使用 tokio::spawn() 方法创建一个新的异步任务,每个新的连接都将创建一个任务,以避免阻塞主循环。在异步任务中,我们使用 tokio::io::AsyncReadExt 和 tokio::io::AsyncWriteExt trait 来实现异步读取和写入数据。我们使用一个循环来持续从客户端读取数据,并将读取的数据写回客户端。如果读取到的字节数为 0,则表示客户端已经关闭连接,退出循环。

最后,我们在异步任务完成后打印一条消息并返回 Ok(())。

下面是对代码的分析:

引入了 tokio 库的 net 和 io 模块:

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

定义了一个 async 函数 main,它是程序的入口函数。由于 main 函数是异步函数,因此在函数签名中加上了 async 关键字,并使用 tokio::main 宏来启动 tokio 运行时。

#[tokio::main]
async fn main() -> Result<(), Box<dyn std::error::Error>> {
    // ...
}

调用 TcpListener::bind 方法创建一个 TcpListener 实例,并将其绑定到本地的 8080 端口上。

let listener = TcpListener::bind("127.0.0.1:8080").await?;

在控制台输出信息,表示服务器已经开始监听连接请求。

println!("Server listening on port 8080");

进入一个无限循环,等待客户端连接请求。当有新的连接请求到来时,调用 listener.accept() 方法,该方法会返回一个包含新连接的 TcpStream 实例以及一个表示远程地址的元组(由于我们不需要远程地址,因此用 _ 占位符来表示)。

loop {
    let (mut socket, _) = listener.accept().await?;
    // ...
}

为每个新连接创建一个新的任务。tokio::spawn 方法接受一个 Future 对象,并在一个新的任务中异步执行该 Future。在这里,我们将一个闭包作为 Future 对象,并将它传递给 tokio::spawn 方法。

tokio::spawn(async move {
    // ...
});

在任务中定义一个可变的 buffer 数组,用于存储从客户端读取的数据。

let mut buffer = [0; 1024];

进入一个无限循环,不断从客户端读取数据,直到客户端关闭连接。在循环中,使用 socket.read() 方法异步读取数据,并使用 match 匹配读取结果。如果读取成功并且读取的字节数为 0,则表示客户端已经关闭连接,跳出循环。如果读取成功并且读取的字节数不为 0,则将读取到的数据写回客户端,使用 socket.write_all() 方法将数据异步写回。

 loop {
   let n = match socket.read(&mut buffer).await {
     Ok(n) if n == 0 => break,
       Ok(n) => n,
         Err(e) => {
         eprintln!("failed to read from socket; err = {:?}", e);
         break;
       }
   };

   if let Err(e) = socket.write_all(&buffer[0..n]).await {
     eprintln!("failed to write to socket; err = {:?}", e);
     break;
   }
 }

要运行该示例程序,我们需要在控制台中执行以下命令:

 cargo run

该程序将会开始监听本地的 8080 端口,当有客户端连接时,会返回 "Server listening on port 8080" 消息。当客户端发送数据时,程序将会将接收到的数据返回给客户端,直到客户端关闭连接。

在本文中,我们了解了如何使用 Tokio 来构建一个基本的 TCP 服务器。通过使用 Tokio 提供的异步 IO 操作,我们可以实现高效的网络应用程序。

相关推荐

MySQL修改密码_mysql怎么改密码忘了怎么办

拥有原来的用户名账户的密码mysqladmin-uroot-ppassword"test123"Enterpassword:【输入原来的密码】忘记原来root密码第一...

数据库密码配置项都不加密?心也太大了吧!

先看一份典型的配置文件...省略...##配置MySQL数据库连接spring.datasource.driver-class-name=com.mysql.jdbc.Driverspr...

Linux基础知识_linux基础入门知识

系统目录结构/bin:命令和应用程序。/boot:这里存放的是启动Linux时使用的一些核心文件,包括一些连接文件以及镜像文件。/dev:dev是Device(设备)的缩写,该目录...

MySQL密码重置_mysql密码重置教程

之前由于修改MySQL加密模式为mysql_native_password时操作失误,导致无法登陆MySQL数据库,后来摸索了一下,对MySQL数据库密码进行重置后顺利解决,步骤如下:1.先停止MyS...

Mysql8忘记密码/重置密码_mysql密码忘了怎么办?

Mysql8忘记密码/重置密码UBUNTU下Mysql8忘记密码/重置密码步骤如下:先说下大概步骤:修改配置文件,使得用空密码可以进入mysql。然后置当前root用户为空密码。再次修改配置文件,不能...

MySQL忘记密码怎么办?Windows环境下MySQL密码重置图文教程

有不少小白在使用Windows进行搭建主机的时候,安装了一些环境后,其中有MySQL设置后,然后不少马大哈忘记了MySQL的密码,导致在一些程序安装及配置的时候无法进行。这个时候怎么办呢?重置密码呗?...

10种常见的MySQL错误,你可中招?_mysql常见错误提示及解决方法

【51CTO.com快译】如果未能对MySQL8进行恰当的配置,您非但可能遇到无法顺利访问、或调用MySQL的窘境,而且还可能给真实的应用生产环境带来巨大的影响。本文列举了十种MySQL...

Mysql解压版安装过程_mysql解压版安装步骤

Mysql是目前软件开发中使用最多的关系型数据库,具体安装步骤如下:第一步:Mysql官网下载最新版(mysql解压版(mysql-5.7.17-winx64)),Mysql官方下载地址为:https...

MySQL Root密码重置指南:Windows新手友好教程

如果你忘记了MySQLroot密码,请按照以下简单步骤进行重置。你需要准备的工具:已安装的MySQL以管理员身份访问命令提示符一点复制粘贴的能力分步操作指南1.创建密码重置文件以管理员...

安卓手机基于python3搜索引擎_python调用安卓so库

环境:安卓手机手机品牌:vivox9s4G运行内存手机软件:utermux环境安装:1.java环境的安装2.redis环境的安装aptinstallredis3.elasticsearch环...

Python 包管理 3 - poetry_python community包

Poetry是一款现代化的Python依赖管理和打包工具。它通过一个pyproject.toml文件来统一管理你的项目依赖、配置和元数据,并用一个poetry.lock文件来锁定所有依赖的精...

Python web在线服务生产环境真实部署方案,可直接用

各位志同道合的朋友大家好,我是一个一直在一线互联网踩坑十余年的编码爱好者,现在将我们的各种经验以及架构实战分享出来,如果大家喜欢,就关注我,一起将技术学深学透,我会每一篇分享结束都会预告下一专题最近经...

官方玩梗:Python 3.14(πthon)稳定版发布,正式支持自由线程

IT之家10月7日消息,当地时间10月7日,Python软件基金会宣布Python3.14.0正式发布,也就是用户期待已久的圆周率(约3.14)版本,再加上谐音梗可戏称为π...

第一篇:如何使用 uv 创建 Python 虚拟环境

想象一下,你有一个使用Python3.10的后端应用程序,系统全局安装了a2.1、b2.2和c2.3这些包。一切运行正常,直到你开始一个新项目,它也使用Python3.10,但需要...

我用 Python 写了个自动整理下载目录的工具

经常用电脑的一定会遇到这种情况:每天我们都在从浏览器、微信、钉钉里下各种文件,什么截图、合同、安装包、临时文档,全都堆在下载文件夹里。起初还想着“过两天再整理”,结果一放就是好几年。结果某天想找一个发...