青少年编程与数学 02-019 Rust 编程基础 23课题、web服务器

青少年编程与数学 02-019 Rust 编程基础 23课题、web服务器

  • 一、单线程Web 服务器
    • 基本实现步骤
    • 完整代码示例
    • 运行结果
    • 项目结构
    • 注意事项
    • 扩展方向
  • 二、多线程Web服务器
    • 1. 基本架构设计
    • 2. 完整实现代码
      • 项目文件结构
      • 文件内容
        • `Cargo.toml`
        • `src/main.rs`
        • `src/lib.rs`
        • `static/hello.html`
        • `static/404.html`
      • 运行项目
      • 说明
    • 3. 关键组件详解
      • 3.1 线程池实现
      • 3.2 工作线程
      • 3.3 请求处理
    • 4. 性能优化方案
      • 4.1 使用更高效的线程池实现
      • 4.2 异步I/O支持
    • 5. 测试服务器性能
    • 6. 生产环境建议
  • 总结

课题摘要:
本文通过创建简单的Web服务器,来了解Rust的Web服务器编程。

关键词:Web服务器、单线程、多线程


一、单线程Web 服务器

下面是一个简单的单线程Web服务器的实现步骤和代码示例。这个服务器可以处理HTTP请求并返回简单的响应。

基本实现步骤

  1. 监听TCP连接
  2. 解析HTTP请求
  3. 构建HTTP响应
  4. 返回响应给客户端

完整代码示例

use std::{fs,io::{prelude::*, BufReader},net::{TcpListener, TcpStream},
};fn main() {let listener = TcpListener::bind("127.0.0.1:8080").unwrap();let addr = listener.local_addr().unwrap();println!("Server running at http://{}", addr);for stream in listener.incoming() {let stream = stream.unwrap();handle_connection(stream);}
}fn handle_connection(mut stream: TcpStream) {let buf_reader = BufReader::new(&stream);let request_line = buf_reader.lines().next().unwrap().unwrap();let (status_line, filename) = if request_line == "GET / HTTP/1.1" {("HTTP/1.1 200 OK", "p19_1s/static/hello.html")} else {("HTTP/1.1 404 NOT FOUND", "p19_1s/static/404.html")};let contents = match fs::read_to_string(filename) {Ok(contents) => contents,Err(e) => {eprintln!("Failed to read file {}: {}", filename, e);return; // 如果文件读取失败,直接返回}};let length = contents.len();let response = format!("{status_line}\r\nContent-Length: {length}\r\n\r\n{contents}");stream.write_all(response.as_bytes()).unwrap();
}
<!DOCTYPE html>
<html lang="en">
<head><meta charset="utf-8"><title>Hello!</title>
</head>
<body><h1>Hello! 青少年编程与数学  02-019 Rust 编程基础</h1>
<p>23课题、web服务器</p>
</body>
</html>

运行结果

Server running at http://127.0.0.1:8080
//点击此链接,可显示网页

项目结构

对于更正式的项目,你可以这样组织文件结构:

webserver/
├── Cargo.toml
├── src/
│   └── main.rs
└── static/├── hello.html└── 404.html

注意事项

  1. 这个服务器是单线程的,一次只能处理一个请求
  2. 没有实现完整的HTTP协议,只处理了基本的GET请求
  3. 错误处理比较简单,生产环境需要更完善的错误处理
  4. 对于生产环境,建议使用成熟的Web框架如Actix-web或Rocket

扩展方向

如果你想扩展这个服务器,可以考虑:

  1. 添加多线程支持(使用线程池)
  2. 实现更完整的HTTP协议解析
  3. 添加路由系统
  4. 支持静态文件服务
  5. 添加中间件支持

这个简单的实现可以帮助你理解Web服务器的基本原理,但在实际项目中,建议使用现有的成熟框架。

二、多线程Web服务器

下面我将详细介绍如何使用Rust构建一个多线程Web服务器,包括线程池的实现、请求处理和性能优化。

1. 基本架构设计

多线程Web服务器的核心组件包括:

  • 主线程:监听TCP连接
  • 线程池:处理传入的连接
  • 共享状态:必要时共享数据

2. 完整实现代码

根据你提供的网页内容,以下是一个完整的 Rust 多线程 Web Server 示例程序。这个程序将使用一个线程池来处理多个请求,避免因单个慢请求而阻塞其他请求。

项目文件结构

hello/
├── Cargo.toml
├── src/
│   ├── main.rs
│   └── lib.rs
├── static/
│   ├── hello.html
│   └── 404.html

文件内容

Cargo.toml
[package]
name = "hello"
version = "0.1.0"
edition = "2024"[dependencies]
src/main.rs
use p19_m::ThreadPool;
use std::{fs,io::{prelude::*, BufReader},net::{TcpListener, TcpStream},thread,time::Duration,
};fn main() {let listener = TcpListener::bind("127.0.0.1:8080").unwrap();let pool = ThreadPool::new(4);println!("Server running at http://127.0.0.1:8080");for stream in listener.incoming() {let stream = stream.unwrap();pool.execute(|| {handle_connection(stream);});}
}fn handle_connection(mut stream: TcpStream) {let buf_reader = BufReader::new(&stream);let request_line = buf_reader.lines().next().unwrap().unwrap();let (status_line, filename) = match &request_line[..] {"GET / HTTP/1.1" => ("HTTP/1.1 200 OK", "p19_2m/static/hello.html"),"GET /sleep HTTP/1.1" => {thread::sleep(Duration::from_secs(5));("HTTP/1.1 200 OK", "p19_2m/static/hello.html")}_ => ("HTTP/1.1 404 NOT FOUND", "p19_2m/static/404.html"),};let contents = fs::read_to_string(filename).unwrap();let length = contents.len();let response = format!("{status_line}\r\nContent-Length: {length}\r\n\r\n{contents}");stream.write_all(response.as_bytes()).unwrap();
}
src/lib.rs
use std::{sync::{mpsc, Arc, Mutex},thread,
};pub struct ThreadPool {workers: Vec<Worker>,sender: mpsc::Sender<Job>,
}type Job = Box<dyn FnOnce() + Send + 'static>;impl ThreadPool {/// Create a new ThreadPool.////// The size is the number of threads in the pool.////// # Panics////// The `new` function will panic if the size is zero.pub fn new(size: usize) -> ThreadPool {assert!(size > 0);let (sender, receiver) = mpsc::channel();let receiver = Arc::new(Mutex::new(receiver));let mut workers = Vec::with_capacity(size);for id in 0..size {workers.push(Worker::new(id, Arc::clone(&receiver)));}ThreadPool { workers, sender }}pub fn execute<F>(&self, f: F)whereF: FnOnce() + Send + 'static,{let job = Box::new(f);self.sender.send(job).unwrap();}
}impl Drop for ThreadPool {fn drop(&mut self) {for worker in &mut self.workers {println!("Shutting down worker {}", worker.id);if let Some(thread) = worker.thread.take() {if let Ok(_) = thread.join() {println!("Worker {} successfully shut down", worker.id);}}}}
}struct Worker {id: usize,thread: Option<thread::JoinHandle<()>>,
}impl Worker {fn new(id: usize, receiver: Arc<Mutex<mpsc::Receiver<Job>>>) -> Worker {let thread = Some(thread::spawn(move || loop {let job = receiver.lock().unwrap().recv().unwrap();println!("Worker {id} got a job; executing.");job();}));Worker { id, thread }}
}
static/hello.html
<!DOCTYPE html>
<html lang="en">
<head><meta charset="utf-8"><title>Hello!</title>
</head>
<body><h1>Hello! 青少年编程与数学  02-019 Rust 编程基础</h1>
<p>23课题、web服务器</p>
</body>
</html>
static/404.html
<!DOCTYPE html>
<html lang="en"><head><meta charset="utf-8"><title>404 Not Found</title></head><body><h1>Oops!</h1><p>Sorry, I don't know what you're asking for.</p></body>
</html>

运行项目

  1. 创建项目文件夹和文件
    根据上述文件结构创建文件夹和文件,并将相应内容填入。

  2. 运行项目

    cargo run
    
  3. 测试 Web Server

    • 打开浏览器,访问 http://127.0.0.1:8080,应该会看到 static/hello.html 的内容。
    • 访问 http://127.0.0.1:8080/sleep,然后在另一个标签页访问 http://127.0.0.1:8080,应该会看到 static/hello.html 的内容,而不会被阻塞。
    • 访问 http://127.0.0.1:8080/something-else,应该会看到 static/404.html 的内容。

说明

  1. 线程池

    • ThreadPool 使用 mpsc::channel 来传递任务。
    • 每个 Worker 线程从通道中接收任务并执行。
  2. 任务分发

    • execute 方法将任务发送到通道中。
    • 每个 Worker 线程循环等待任务并执行。
  3. 错误处理

    • 使用 unwrap 来简化代码,但在生产环境中应进行更健壮的错误处理。
  4. 控制台提示

    • 服务器启动后,会在控制台打印 Server running at http://127.0.0.1:8080,方便用户访问。

这个多线程 Web Server 示例程序可以同时处理多个请求,避免因单个慢请求而阻塞其他请求。

3. 关键组件详解

3.1 线程池实现

线程池的核心是使用通道(mpsc)进行任务分发:

struct ThreadPool {workers: Vec<Worker>,sender: mpsc::Sender<Job>,
}type Job = Box<dyn FnOnce() + Send + 'static>;
  • workers 存储工作线程
  • sender 用于发送任务到工作线程
  • Job 是类型别名,表示可在线程间传递的闭包

3.2 工作线程

struct Worker {id: usize,thread: thread::JoinHandle<()>,
}impl Worker {fn new(id: usize, receiver: Arc<Mutex<mpsc::Receiver<Job>>>) -> Worker {let thread = thread::spawn(move || loop {let job = receiver.lock().unwrap().recv().unwrap();println!("Worker {} got a job; executing.", id);job();});Worker { id, thread }}
}

每个工作线程不断从接收端获取任务并执行。

3.3 请求处理

fn handle_connection(mut stream: TcpStream) {// ...解析请求...// 模拟耗时操作if request_line == "GET /sleep HTTP/1.1" {thread::sleep(Duration::from_secs(5));}// ...构建响应...
}

4. 性能优化方案

4.1 使用更高效的线程池实现

可以使用现成的线程池库如rayon

[dependencies]
rayon = "1.5"

然后修改主函数:

use rayon::ThreadPoolBuilder;fn main() {let pool = ThreadPoolBuilder::new().num_threads(num_cpus::get()).build().unwrap();let listener = TcpListener::bind("127.0.0.1:8080").unwrap();for stream in listener.incoming() {let stream = stream.unwrap();pool.spawn(|| handle_connection(stream));}
}

4.2 异步I/O支持

对于更高性能的场景,可以使用异步运行时如tokio

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

异步版本实现:

use tokio::{io::{AsyncReadExt, AsyncWriteExt},net::{TcpListener, TcpStream},
};#[tokio::main]
async fn main() -> Result<(), Box<dyn std::error::Error>> {let listener = TcpListener::bind("127.0.0.1:8080").await?;loop {let (mut stream, _) = listener.accept().await?;tokio::spawn(async move {let mut buf = [0; 1024];let n = stream.read(&mut buf).await.unwrap();let response = if buf.starts_with(b"GET / HTTP/1.1") {"HTTP/1.1 200 OK\r\n\r\nHello, World!"} else {"HTTP/1.1 404 NOT FOUND\r\n\r\n"};stream.write_all(response.as_bytes()).await.unwrap();});}
}

5. 测试服务器性能

可以使用wrk工具测试服务器性能:

# 测试基础性能
wrk -t12 -c400 -d30s http://127.0.0.1:8080/# 测试长连接性能
wrk -t12 -c400 -d30s -H "Connection: keep-alive" http://127.0.0.1:8080/

6. 生产环境建议

  1. 使用成熟框架:如Actix-web、Rocket或Warp
  2. 配置优化
    • 调整线程池大小
    • 设置合理的backlog
    • 启用TCP_NODELAY
  3. 安全考虑
    • 请求超时设置
    • 请求大小限制
    • 防止DDoS攻击

总结

以上,介绍了如何使用Rust语言构建单线程和多线程Web服务器。首先,通过一个简单的单线程服务器示例,展示了如何监听TCP连接、解析HTTP请求并返回响应。该服务器能够处理基本的GET请求,但存在一次只能处理一个请求的限制。接着,文章深入探讨了多线程Web服务器的实现,包括线程池的设计与实现、请求处理逻辑以及性能优化方案。通过使用线程池,服务器能够并发处理多个请求,显著提升了性能。文章还介绍了如何使用现成的线程池库(如rayon)和异步运行时(如tokio)来进一步优化服务器性能。此外,还提供了测试服务器性能的方法和生产环境下的优化建议。最后,文章提出了完整的项目结构,为构建更复杂的Web服务器提供了参考。

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.pswp.cn/diannao/84731.shtml

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

(14)JVM弹性内存管理

文章目录 &#x1f680; JVM弹性内存管理&#xff1a;K8s环境下的内存优化终极攻略⚡ TL;DR&#x1f635; 等等&#xff0c;为什么我需要关心这个&#xff1f;&#x1f6e0;️ 五步搞定弹性内存&#xff08;拯救你的Java应用&#xff09;1️⃣ JVM参数调教2️⃣ 监控指标全覆盖…

Spring Boot集成Spring AI与Milvus实现智能问答系统

在Spring Boot中集成Spring AI与Milvus实现智能问答系统 引言 随着人工智能技术的快速发展&#xff0c;智能问答系统在企业中的应用越来越广泛。然而&#xff0c;传统的问答系统往往面临AI幻觉&#xff08;Hallucination&#xff09;问题&#xff0c;即生成不准确或无意义的回…

电脑网络如何改ip地址?ip地址改不了怎么回事

在日常使用电脑上网时&#xff0c;我们有时会遇到需要更改IP地址的情况&#xff0c;比如访问某些受限制的网站、解决网络冲突问题&#xff0c;或者出于隐私保护的需求。然而&#xff0c;许多用户在尝试修改IP地址时可能会遇到各种问题&#xff0c;例如IP地址无法更改、修改后无…

SQL进阶之旅 Day 1:高效表设计与规范

SQL进阶之旅 Day 1&#xff1a;高效表设计与规范 开篇 欢迎来到为期30天的“SQL进阶之旅”系列的第一天&#xff01;今天我们将从数据库表设计的基础入手&#xff0c;讨论如何通过合理的表设计来提升数据库性能。这不仅是每位数据库开发工程师的基本功&#xff0c;也是解决实…

【MySQL】第11节|MySQL 8.0 主从复制原理分析与实战

一、MySQL主从复制基础 1. 核心概念 定义&#xff1a; MySQL主从复制是将主库&#xff08;Source/Master&#xff09;的数据变更同步到一个或多个从库&#xff08;Replica/Slave&#xff09;的机制&#xff0c;默认采用异步复制&#xff0c;支持全库、指定库或表的同步。 角…

怎么判断一个Android APP使用了Cocos 这个跨端框架

要判断一个 Android 应用是否使用了 Cocos 跨端框架&#xff0c;可以通过以下步骤进行验证&#xff1a; 一、安装包结构分析 1. 解压 APK 将 .apk 文件重命名为 .zip 并解压&#xff0c;检查以下特征文件&#xff1a; • lib/ 目录&#xff1a; Cocos 引擎的核心原生库文件通常…

删除word中由奇偶页和页码1设置多出来的空白页

问题&#xff1a; 在调整毕设论文格式时&#xff0c;要求奇偶页眉设置不同&#xff0c;且摘要页的页码是1&#xff08;I&#xff09;。如果摘要页在整个文档的第偶数页&#xff0c;将其页码设置为1后会变为奇数页&#xff0c;word为了凑齐奇偶页&#xff0c;会在摘要前增加一个…

# 探索自然语言处理的奥秘:基于 Qwen 模型的文本分类与对话系统实现

探索自然语言处理的奥秘&#xff1a;基于 Qwen 模型的文本分类与对话系统实现 在当今数字化时代&#xff0c;自然语言处理&#xff08;NLP&#xff09;技术正以前所未有的速度改变着我们的生活和工作方式。从智能语音助手到自动文本生成&#xff0c;从情感分析到机器翻译&…

Linux Shell 切换

在 Linux 系统中&#xff0c;切换至 Bash Shell 在 Linux 系统中&#xff0c;切换至 Bash Shell 的方法如下&#xff1a; 临时切换到 Bash 直接在终端输入以下命令&#xff0c;启动一个新的 Bash 会话&#xff1a; bash 退出时输入 exit 或按 CtrlD 返回原 Shell。 永久切换…

在Windows上,将 Ubuntu WSL 安装并迁移到 D 盘完整教程(含 Appx 安装与迁移导入)

&#x1f4bb; 将 Ubuntu WSL 安装并迁移到 D 盘完整教程&#xff08;含 Appx 安装与迁移导入&#xff09; 本文记录如何在 Windows 系统中手动启用 WSL、下载 Ubuntu 安装包、安装并迁移 Ubuntu 到 D 盘&#xff0c;避免默认写入 C 盘&#xff0c;提高系统性能与可维护性。 ✅…

doucker 挂载卷

在 Docker 中&#xff0c;挂载卷&#xff08;Volumes&#xff09;是一种非常重要的功能&#xff0c;它允许你将宿主机的文件系统与容器的文件系统进行共享。挂载卷不仅可以用于持久化数据&#xff0c;还可以用于在宿主机和容器之间传递文件。 挂载卷的类型 Docker 支持多种类型…

BLIP3-o:一系列完全开源的统一多模态模型——架构、训练与数据集

摘要 在近期关于多模态模型的研究中&#xff0c;将图像理解与生成统一起来受到了越来越多的关注。尽管图像理解的设计选择已经得到了广泛研究&#xff0c;但对于具有图像生成功能的统一框架而言&#xff0c;其最优模型架构和训练方案仍有待进一步探索。鉴于自回归和扩散模型在…

数据分析案例-基于红米和华为手机的用户评论分析

&#x1f935;‍♂️ 个人主页&#xff1a;艾派森的个人主页 ✍&#x1f3fb;作者简介&#xff1a;Python学习者 &#x1f40b; 希望大家多多支持&#xff0c;我们一起进步&#xff01;&#x1f604; 如果文章对你有帮助的话&#xff0c; 欢迎评论 &#x1f4ac;点赞&#x1f4…

基础框架 兼容视频格式

基础框架 兼容视频格式 修改 \src\components\Upload\src\BasicUpload.vue 数据库新增 vue <template><div class"w-full"><div class"upload"><div class"upload-card"><!--图片列表--><divclass"uploa…

qiankun 子应用怎样通过 props拿到子应用【注册之后挂载之前】主应用中发生变更的数据

场景描述&#xff1a;子应用需要在接口调用和页面渲染时&#xff0c;需要用到主应用登录之后拿到的用户数据 逻辑前提&#xff1a; 1、主应用在 main.js中通过 registerMicroApps注册了子应用 2、主应用登录之后将用户数据传递给子应用 >> 原先的做法&#xff08;有问题&…

Hooks 进阶:自定义 Hook 的设计与实践

引言 React Hooks 已成为现代 React 开发的核心范式&#xff0c;而自定义 Hook 则为我们提供了强大的代码复用机制。 自定义 Hook 的基础原理 自定义 Hook 本质上是一种函数复用机制&#xff0c;它允许我们将组件逻辑提取到可重用的函数中。与传统的高阶组件(HOC)和 render …

锂电电动扭剪扳手市场报告:现状、趋势与竞争格局深度解析

一、锂电电动扭剪扳手市场概述 锂电电动扭剪扳手作为建筑施工、钢结构安装等领域的关键工具&#xff0c;凭借其便携性、高效性及环保特性&#xff0c;正逐步替代传统手动及气动工具。该设备通过锂电池供电&#xff0c;结合智能扭矩控制技术&#xff0c;可精准完成高强度螺栓的…

[面试精选] 0076. 最小覆盖子串

文章目录 1. 题目链接2. 题目描述3. 题目示例4. 解题思路5. 题解代码6. 复杂度分析 1. 题目链接 76. 最小覆盖子串 - 力扣&#xff08;LeetCode&#xff09; 2. 题目描述 给你一个字符串 s 、一个字符串 t 。返回 s 中涵盖 t 所有字符的最小子串。如果 s 中不存在涵盖 t 所有字…

rabbitmq的高级特性

一.发送者的可靠性 1.生产者重试机制 修改publisher模块的application.yaml文件 spring:rabbitmq:connection-timeout: 1s # 设置MQ的连接超时时间template:retry:enabled: true # 开启超时重试机制initial-interval: 1000ms # 失败后的初始等待时间multiplier: 1 # 失败后下…

北京大学肖臻老师《区块链技术与应用》公开课:02-BTC-密码学原理

文章目录 1.比特币中用到的密码学的功能2. hash3. 签名 1.比特币中用到的密码学的功能 比特币中用到密码学中两个功能&#xff1a; hash、 签名。 2. hash hash函数的三个特性&#xff1a;抗碰撞性&#xff08;Collision Resistance&#xff09;、隐蔽性&#xff08;Hiding&…