Actix-webRust Web框架入门教程

文章目录

    • 引言
    • Actix-web是什么?
    • 准备工作
    • 你的第一个Actix-web应用
    • 理解代码结构
    • 处理请求和响应
      • 接收请求数据
      • 返回响应
    • 中间件 - 增强你的应用
    • 状态管理和依赖注入
    • 实用示例:构建RESTful API
    • 测试你的Actix-web应用
    • 部署Actix-web应用
    • 结语
    • 额外资源

引言

嘿,各位开发者们!今天我要和大家分享一个在Rust生态系统中备受瞩目的Web框架——Actix-web。作为Rust世界中最受欢迎的Web框架之一,Actix-web以其惊人的性能和友好的API设计赢得了众多开发者的青睐。无论你是Rust新手还是老手,这篇教程都将帮助你快速上手这个强大的工具!

我第一次接触Actix-web时简直被它的速度震惊了(真的超级快!!!)。如果你正在寻找一个高性能、类型安全且易于使用的Web框架,那么Actix-web绝对值得你花时间学习。

Actix-web是什么?

Actix-web是一个用Rust编写的高性能Web框架,它建立在actor系统Actix的基础上(虽然现在已经可以独立使用)。它提供了创建HTTP服务器和客户端的工具,支持WebSockets、HTTP/2等现代Web技术。

与其他Web框架相比,Actix-web的主要优势在于:

  • 极致性能 - 在多项基准测试中,它经常位居前列(有时甚至超过Go和Node.js框架)
  • 类型安全 - 利用Rust的类型系统,避免常见错误
  • 异步支持 - 完全拥抱Rust的异步编程模型
  • 灵活性 - 从简单API到复杂应用,都能胜任

准备工作

在开始之前,你需要准备好Rust开发环境。如果你还没有安装Rust,可以通过rustup来安装:

curl --proto '=https' --tlsv1.2 -sSf https://sh.rustup.rs | sh

确保你的Rust是最新版本:

rustup update

接下来,创建一个新的Rust项目:

cargo new actix_demo
cd actix_demo

然后,在Cargo.toml文件中添加Actix-web依赖:

[dependencies]
actix-web = "4.3.1"

你的第一个Actix-web应用

让我们开始创建一个简单的"Hello World"应用。打开src/main.rs文件,并替换为以下内容:

use actix_web::{get, post, web, App, HttpResponse, HttpServer, Responder};#[get("/")]
async fn hello() -> impl Responder {HttpResponse::Ok().body("Hello, Actix-web!")
}#[post("/echo")]
async fn echo(req_body: String) -> impl Responder {HttpResponse::Ok().body(req_body)
}async fn manual_hello() -> impl Responder {HttpResponse::Ok().body("Hey there!")
}#[actix_web::main]
async fn main() -> std::io::Result<()> {HttpServer::new(|| {App::new().service(hello).service(echo).route("/hey", web::get().to(manual_hello))}).bind(("127.0.0.1", 8080))?.run().await
}

运行这个程序:

cargo run

现在你可以访问 http://localhost:8080/ 看到"Hello, Actix-web!"的响应。也可以通过POST请求发送数据到 http://localhost:8080/echo 端点,或者访问 http://localhost:8080/hey 查看另一个响应。

理解代码结构

让我们分解上面的代码,以便理解Actix-web的核心概念:

  1. 路由处理函数:在Actix-web中,每个路由对应一个处理函数,如上例中的helloechomanual_hello。这些函数必须是异步的,并且返回实现了Responder特质的类型。

  2. 宏注解#[get("/")]#[post("/echo")]是路由宏,它们简化了路由定义。也可以通过.route()方法手动定义路由,就像/hey路由那样。

  3. HttpServer和AppHttpServer负责处理HTTP请求,而App是应用程序的主体,它包含了所有的路由和中间件配置。

  4. 绑定和运行:使用.bind()方法指定服务器监听的地址和端口,然后调用.run().await启动服务器。

处理请求和响应

在Web开发中,处理请求和生成响应是最基本的任务。Actix-web提供了多种方式来处理这些操作。

接收请求数据

Actix-web支持多种方式获取请求数据:

  1. 路径参数
#[get("/users/{id}")]
async fn get_user(path: web::Path<(u32,)>) -> impl Responder {let user_id = path.into_inner().0;HttpResponse::Ok().body(format!("User ID: {}", user_id))
}
  1. 查询参数
#[derive(Deserialize)]
struct Info {name: String,age: u32,
}#[get("/info")]
async fn info(query: web::Query<Info>) -> impl Responder {format!("Welcome {}! You are {} years old.", query.name, query.age)
}
  1. JSON请求体
#[derive(Deserialize)]
struct User {name: String,email: String,
}#[post("/users")]
async fn create_user(user: web::Json<User>) -> impl Responder {HttpResponse::Created().json(user.into_inner())
}

返回响应

Actix-web提供了多种方式来构建响应:

  1. 简单文本响应
HttpResponse::Ok().body("Hello world!")
  1. JSON响应
#[derive(Serialize)]
struct Response {message: String,status: String,
}async fn json_response() -> impl Responder {let response = Response {message: "Success".to_string(),status: "OK".to_string(),};web::Json(response)
}
  1. 重定向
async fn redirect() -> impl Responder {HttpResponse::Found().append_header(("Location", "/login")).finish()
}

中间件 - 增强你的应用

中间件允许你在请求处理的前后添加额外的逻辑。Actix-web提供了许多内置的中间件,如日志记录、压缩、CORS支持等。你也可以创建自定义中间件。

以下是如何添加日志中间件的示例:

use actix_web::middleware::Logger;#[actix_web::main]
async fn main() -> std::io::Result<()> {std::env::set_var("RUST_LOG", "actix_web=info");env_logger::init();HttpServer::new(|| {App::new().wrap(Logger::default()).service(hello)}).bind(("127.0.0.1", 8080))?.run().await
}

这会记录所有传入的请求,包括路径、状态码和处理时间。

状态管理和依赖注入

在Web应用中,你经常需要在不同的请求处理程序之间共享状态(如数据库连接)。Actix-web提供了一种类型安全的方式来实现这一点:

use std::sync::Mutex;struct AppState {counter: Mutex<i32>,
}#[get("/count")]
async fn count(data: web::Data<AppState>) -> String {let mut counter = data.counter.lock().unwrap();*counter += 1;format!("Request number: {}", counter)
}#[actix_web::main]
async fn main() -> std::io::Result<()> {let app_state = web::Data::new(AppState {counter: Mutex::new(0),});HttpServer::new(move || {App::new().app_data(app_state.clone()).service(count)}).bind(("127.0.0.1", 8080))?.run().await
}

这个例子创建了一个简单的计数器,每次访问/count路径时增加计数。

实用示例:构建RESTful API

让我们构建一个简单的待办事项API,它支持创建、读取、更新和删除操作(CRUD):

use actix_web::{web, App, HttpResponse, HttpServer, Responder};
use serde::{Deserialize, Serialize};
use std::sync::Mutex;#[derive(Debug, Serialize, Deserialize, Clone)]
struct Task {id: Option<u32>,title: String,completed: bool,
}struct AppState {tasks: Mutex<Vec<Task>>,counter: Mutex<u32>,
}async fn get_tasks(data: web::Data<AppState>) -> impl Responder {let tasks = data.tasks.lock().unwrap();HttpResponse::Ok().json(&*tasks)
}async fn get_task_by_id(path: web::Path<u32>, data: web::Data<AppState>) -> impl Responder {let task_id = path.into_inner();let tasks = data.tasks.lock().unwrap();if let Some(task) = tasks.iter().find(|t| t.id == Some(task_id)) {HttpResponse::Ok().json(task)} else {HttpResponse::NotFound().body("Task not found")}
}async fn create_task(task: web::Json<Task>, data: web::Data<AppState>) -> impl Responder {let mut tasks = data.tasks.lock().unwrap();let mut counter = data.counter.lock().unwrap();let mut new_task = task.into_inner();*counter += 1;new_task.id = Some(*counter);tasks.push(new_task.clone());HttpResponse::Created().json(new_task)
}async fn update_task(path: web::Path<u32>, task: web::Json<Task>, data: web::Data<AppState>) -> impl Responder {let task_id = path.into_inner();let mut tasks = data.tasks.lock().unwrap();if let Some(pos) = tasks.iter().position(|t| t.id == Some(task_id)) {let mut updated_task = task.into_inner();updated_task.id = Some(task_id);tasks[pos] = updated_task.clone();HttpResponse::Ok().json(updated_task)} else {HttpResponse::NotFound().body("Task not found")}
}async fn delete_task(path: web::Path<u32>, data: web::Data<AppState>) -> impl Responder {let task_id = path.into_inner();let mut tasks = data.tasks.lock().unwrap();if let Some(pos) = tasks.iter().position(|t| t.id == Some(task_id)) {tasks.remove(pos);HttpResponse::Ok().body("Task deleted")} else {HttpResponse::NotFound().body("Task not found")}
}#[actix_web::main]
async fn main() -> std::io::Result<()> {let app_state = web::Data::new(AppState {tasks: Mutex::new(Vec::new()),counter: Mutex::new(0),});HttpServer::new(move || {App::new().app_data(app_state.clone()).route("/tasks", web::get().to(get_tasks)).route("/tasks", web::post().to(create_task)).route("/tasks/{id}", web::get().to(get_task_by_id)).route("/tasks/{id}", web::put().to(update_task)).route("/tasks/{id}", web::delete().to(delete_task))}).bind(("127.0.0.1", 8080))?.run().await
}

这个例子实现了一个完整的RESTful API,支持以下操作:

  • GET /tasks - 获取所有任务
  • POST /tasks - 创建新任务
  • GET /tasks/{id} - 获取特定任务
  • PUT /tasks/{id} - 更新特定任务
  • DELETE /tasks/{id} - 删除特定任务

测试你的Actix-web应用

Actix-web提供了测试工具,让你可以方便地测试你的应用:

#[cfg(test)]
mod tests {use super::*;use actix_web::{test, App};#[actix_web::test]async fn test_hello_endpoint() {let app = test::init_service(App::new().service(hello)).await;let req = test::TestRequest::get().uri("/").to_request();let resp = test::call_service(&app, req).await;assert!(resp.status().is_success());let body = test::read_body(resp).await;assert_eq!(body, "Hello, Actix-web!");}
}

这个测试创建了一个测试服务,发送GET请求到"/"路径,然后验证响应状态和内容。

部署Actix-web应用

当你准备部署你的应用时,记住以下几点:

  1. 使用--release标志进行编译,以获得最佳性能:
cargo build --release
  1. 考虑使用反向代理(如Nginx)来处理SSL、负载均衡等:
server {listen 80;server_name example.com;location / {proxy_pass http://localhost:8080;proxy_http_version 1.1;proxy_set_header Upgrade $http_upgrade;proxy_set_header Connection 'upgrade';proxy_set_header Host $host;proxy_cache_bypass $http_upgrade;}
}
  1. 为生产环境配置日志和监控。

结语

Actix-web是一个功能强大且高性能的Web框架,非常适合构建从简单API到复杂Web应用的各种项目。它的类型安全和并发模型使得开发安全可靠的Web服务变得简单。

虽然学习曲线可能比一些其他语言的Web框架稍陡(主要是因为Rust本身),但投入的时间绝对值得。一旦你习惯了Actix-web的开发模式,你会发现它不仅高效而且非常愉快!

希望这篇教程能帮助你开始使用Actix-web!记住,最好的学习方式是实践,所以赶快动手尝试吧。(相信我,你会爱上它的!)

额外资源

  • Actix-web官方文档
  • Actix-web GitHub仓库
  • Rust编程语言官方文档

祝你编码愉快!

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

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

相关文章

若依框架前端通过 nginx docker 镜像本地运行

1. 前言 项目运行过程图&#xff1a;对于前端项目通过命令 npm run build 打包后&#xff0c;无法直接运行。存在如下错误&#xff1a;可以通过配置 nginx 服务器运行前端项目解决如上问题。 2. Nginx 运行 采用 docker 镜像的方式运行&#xff0c;docker-compose.yml 文件内容…

浅聊一下HTTP协议

在日常上网浏览网页、刷视频时&#xff0c;背后都离不开 HTTP 协议的支持。作为 Web 世界的 “交通规则”&#xff0c;它负责服务器和客户端浏览器之间的数据传输。这篇文章就带大家全面了解 HTTP 协议&#xff0c;从基本概念到通信细节&#xff0c;再到安全相关的 HTTPS&#…

机器人控制器开发(定位——cartographer ros2 使用2)

文章总览 1 纯定位模式 当完成建图后&#xff0c;会生成pbstream格式的地图文件 配置纯定位模式的lua脚本 backpack_2d_localization.lua include "backpack_2d.lua"TRAJECTORY_BUILDER.pure_localization_trimmer {max_submaps_to_keep 3, } POSE_GRAPH.optimi…

《大数据之路1》笔记3:数据管理

一 元数据 1.1 元数据概述 定义&#xff1a; 元数据是关于数据的数据&#xff0c;元数据打通了源数据、数据仓库、数据应用&#xff0c;记录了数据从生产到消费的全部过程。元数据主要记录数据仓库中模型的定义、各层级间的映射关系、监控数据仓库的数据状态和ETL的任务运行状态…

排序实现java

排序算法概述Java中实现排序可以通过多种方式&#xff0c;包括内置方法、自定义算法或使用第三方库。常见的排序算法有冒泡排序、选择排序、插入排序、快速排序、归并排序等。使用Arrays.sort()方法对于数组排序&#xff0c;Java提供了Arrays.sort()方法&#xff0c;支持对基本…

51c大模型~合集182

我自己的原文哦~ https://blog.51cto.com/whaosoft/14174587 #LaV-CoT 超越GPT-4o&#xff0c;蚂蚁集团与南洋理工大学提出&#xff1a;首个语言感知的视觉思维链 随着大型视觉语言模型&#xff08;VLM&#xff09;的飞速发展&#xff0c;它们在处理复杂的视…

C++ STL之deque的使用和模拟实现

目录 deque 核心本质与定位 与stack和queue的关系: deque的使用 deque的底层实现 deque的原理介绍 deque的缺陷 总结: deque deque文档 : deque 翻译: 双端队列 deque&#xff08;通常发音类似“deck”&#xff09;是“double-ended queue”&#xff08;双端队列&…

布草洗涤厂设备租赁押金原路退回系统—东方仙盟

设备租赁状态设备管理添加设备设备收押金设备退押金在布草洗涤行业的运营版图中&#xff0c;设备租赁是连接厂商与客户的重要纽带&#xff0c;而押金的收取与退还则是这一环节中关乎信任与效率的关键节点。未来之窗布草洗涤厂深谙此道&#xff0c;专为设备租赁业务打造的 “押金…

换源rocklinux和centos

一、Rockylinux换源&#xff0c;国外的源换成国内的源#nmcli connection modify ens33 ipv4.addresses 192.168.121.11 ipv4.gateway 192.168.121.2 ipv4.method manual ipv4.dns 114.114.114.114 connection.autoconnect yes修改地址#systemctl stop firewalld#systemctl diab…

第一部分:服务器硬件配置

目录1.1 服务器上架与连线1.2 启用CPU虚拟化功能&#xff08;BIOS设置&#xff09;1.3 配置RAID存储步骤1&#xff1a;进入RAID配置界面步骤2&#xff1a;确认RAID控制器信息步骤3&#xff1a;创建系统RAID&#xff08;用于安装ESXi&#xff09;步骤4&#xff1a;创建数据RAID&…

手搓一个 DELL EMC Unity存储系统健康检查清单

写在前面对于DELL EMC存储系统Unity的一些深度的健康检查通过Web的Unisphere图形化界面是做不到的&#xff0c;图形化界面只能看到是否有告警&#xff0c;物理的东西是否有问题的&#xff0c;逻辑的Pool和LUN等是否ready&#xff0c;再深入的潜在的问题是查不到的。另外&#x…

【数据结构】二叉树的概念

01 概念定义&#xff1a;二叉树既然叫二叉树&#xff0c;顾名思义即度最大为2的树称为二叉树。 它的度可以为 1 也可以为 0&#xff0c;但是度最大为 2 。 一颗二叉树是节点的一个有限集合&#xff0c;该集合&#xff1a;① 由一个根节点加上两棵被称为左子树和右子树的二叉树组…

【RK3576】【Android14】如何在Android14下单独编译kernel-6.1?

单独编译kernel依赖如下几个源码&#xff1a;【交叉编译工具链】prebuilts/clang/host/linux-x86/clang-r487747c【内核源码】kernel-6.1为什么Android下编译内核使用clang作为交叉编译工具链而不是GCC&#xff1f;Android 14 选择使用预置的 Clang 工具链&#xff08;如 clang…

什么是Redis的Pipeline

介绍Redis的Pipeline是一种网络优化技术&#xff0c;在没有Pipeline的时候&#xff0c;客户端往redis发送请求&#xff0c;客户端需要等到redis响应之后才能发送下一个请求。而Pipeline&#xff0c;使redis可以一次性接收多个请求。减少了通信次数&#xff0c;显著的提高了性能…

【ElementUI el-table跨页勾选】

一、el-table需加上refs和 row-key属性 二、type"selection"勾选框 需加上 reserve-selection储备选择属性 三、在分页请求数据时&#xff0c;触发 setSelected()方法 四、在 selection-change变化时保存 selectedRows <el-table ref"tables" :data&quo…

论文阅读/博弈论/拍卖:《Truthful Auction for Cooperative Communications》

摘要&#xff1a;一方面&#xff0c;协作通信由于其在提升无线网络容量方面的巨大潜力而日益受到关注。另一方面&#xff0c;协作通信技术的实际应用却很少见&#xff0c;即使在一些对带宽需求极高的应用场景中&#xff0c;系统设计者也并未采用协作通信技术来开发创新的网络解…

系统软中间件:连接软件与硬件的桥梁

理解“系统软中间件”这个术语很重要&#xff0c;它实际上是两个紧密相关但又不同的概念的组合&#xff1a; 系统软件中间件 严格来说&#xff0c;“系统软中间件”不是一个标准的独立术语。它通常指的是属于系统软件范畴的中间件&#xff0c;或者理解为作为系统软件重要组成部…

音视频学习(六十四):avc1 hvc1和hev1

基础概念缩写编码标准FourCC说明AVC/H.264Advanced Video Codingavc1最常用的 H.264 编码标识符&#xff0c;兼容 MP4/MOV/FMP4 等容器。HEVC/H.265High Efficiency Video Codinghvc1HEVC 视频流在 MP4/FMP4 中常用标识符&#xff0c;要求存储 NALU 的 VPS/SPS/PPS&#xff08;…

【WIT】编程百问一

01 什么时postman&#xff1f; Postman 是一款专门用于帮助开发人员处理 API 的工具&#xff0c;它的作用主要有以下几个方面&#xff1a; 方便调试 API&#xff1a;就像你打电话给别人要先拨对号码一样&#xff0c;开发人员要让不同的软件系统之间通过 API 进行通信&#xff…

RAG 从入门到放弃?丐版 demo 实战笔记(go+python)

背景 我当前有一个业务系统&#xff0c;希望能添加一个机器人助手。直接使用大模型&#xff0c;由于缺少相关的业务数据&#xff0c;效果并不理想&#xff0c;了解一下 RAG。 什么是 RAG RAG(Retrieval Augmented Generation)&#xff0c;搜索引擎 大模型。 简单来说就是从…