Rust 同步方式訪問 REST API 的完整指南

Rust 同步方式訪問 REST API 的完整指南

在 Rust 中不使用異步機制訪問 REST API 是完全可行的,特別適合簡單應用、腳本或不需要高并發的場景。以下是完整的同步實現方案:

📦 依賴選擇

推薦庫:

[dependencies]
reqwest = { version = "0.11", features = ["blocking", "json"] }
serde = { version = "1.0", features = ["derive"] }
serde_json = "1.0"

🔧 基礎 GET 請求

1. 簡單文本響應

use reqwest::blocking::get;fn main() -> Result<(), Box<dyn std::error::Error>> {let url = "https://jsonplaceholder.typicode.com/posts/1";let response = get(url)?;if response.status().is_success() {let body = response.text()?;println!("Response body: {}", body);} else {println!("Request failed with status: {}", response.status());}Ok(())
}

2. 解析 JSON 響應

use serde::Deserialize;#[derive(Debug, Deserialize)]
struct Post {userId: u32,id: u32,title: String,body: String,
}fn main() -> Result<(), Box<dyn std::error::Error>> {let url = "https://jsonplaceholder.typicode.com/posts/1";let response = get(url)?;if response.status().is_success() {let post: Post = response.json()?;println!("Post title: {}", post.title);println!("Full post: {:#?}", post);} else {println!("Request failed with status: {}", response.status());}Ok(())
}

📤 POST 請求示例

1. 發送 JSON 數據

use reqwest::blocking::Client;
use serde::Serialize;#[derive(Debug, Serialize)]
struct NewPost {title: String,body: String,userId: u32,
}fn main() -> Result<(), Box<dyn std::error::Error>> {let client = Client::new();let new_post = NewPost {title: "My New Post".to_string(),body: "This is the body of my new post".to_string(),userId: 1,};let response = client.post("https://jsonplaceholder.typicode.com/posts").json(&new_post).send()?;println!("Status: {}", response.status());println!("Response: {}", response.text()?);Ok(())
}

2. 發送表單數據

use reqwest::blocking::Client;fn main() -> Result<(), Box<dyn std::error::Error>> {let client = Client::new();let params = [("username", "john_doe"), ("password", "secret123")];let response = client.post("https://example.com/login").form(&params).send()?;println!("Login response: {}", response.text()?);Ok(())
}

🔐 高級功能

1. 添加請求頭

use reqwest::blocking::Client;
use reqwest::header;fn main() -> Result<(), Box<dyn std::error::Error>> {let client = Client::new();let mut headers = header::HeaderMap::new();headers.insert(header::AUTHORIZATION, "Bearer token123".parse()?);headers.insert(header::USER_AGENT, "MySyncRustClient/1.0".parse()?);let response = client.get("https://api.example.com/protected").headers(headers).send()?;println!("Protected resource: {}", response.text()?);Ok(())
}

2. 超時設置

use reqwest::blocking::Client;
use std::time::Duration;fn main() -> Result<(), Box<dyn std::error::Error>> {let client = Client::builder().timeout(Duration::from_secs(5)).build()?;let response = client.get("https://api.example.com/slow-endpoint").send()?;println!("Response: {}", response.text()?);Ok(())
}

3. 處理分頁

use reqwest::blocking::Client;
use serde::Deserialize;#[derive(Debug, Deserialize)]
struct Page {items: Vec<Item>,next_page: Option<u32>,
}#[derive(Debug, Deserialize)]
struct Item {id: u32,name: String,
}fn fetch_all_items() -> Result<Vec<Item>, Box<dyn std::error::Error>> {let client = Client::new();let mut all_items = Vec::new();let mut page_num = 1;loop {let url = format!("https://api.example.com/items?page={}", page_num);let response = client.get(&url).send()?;if !response.status().is_success() {return Err(format!("Request failed: {}", response.status()).into());}let page: Page = response.json()?;all_items.extend(page.items);match page.next_page {Some(next) => page_num = next,None => break,}}Ok(all_items)
}

🛡? 錯誤處理最佳實踐

1. 自定義錯誤類型

use thiserror::Error;#[derive(Error, Debug)]
enum ApiError {#[error("HTTP request failed: {0}")]HttpError(#[from] reqwest::Error),#[error("API returned error: {0}")]ApiError(String),#[error("Invalid response format")]ParseError,
}fn fetch_data() -> Result<String, ApiError> {let response = reqwest::blocking::get("https://api.example.com/data")?;if response.status().is_success() {response.text().map_err(|_| ApiError::ParseError)} else {let status = response.status();let body = response.text().unwrap_or_default();Err(ApiError::ApiError(format!("{}: {}", status, body)))}
}

2. 重試機制

use reqwest::blocking::Client;
use std::thread;
use std::time::Duration;fn fetch_with_retry(url: &str, max_retries: u32) -> Result<String, Box<dyn std::error::Error>> {let client = Client::new();let mut retries = 0;loop {match client.get(url).send() {Ok(response) if response.status().is_success() => {return response.text().map_err(|e| e.into());}Ok(response) => {eprintln!("Request failed: {}", response.status());}Err(e) => {eprintln!("Request error: {}", e);}}retries += 1;if retries >= max_retries {return Err("Max retries exceeded".into());}// 指數退避let delay = 2u64.pow(retries);eprintln!("Retrying in {} seconds...", delay);thread::sleep(Duration::from_secs(delay));}
}

📊 性能優化

1. 復用 HTTP 客戶端

use reqwest::blocking::Client;fn main() -> Result<(), Box<dyn std::error::Error>> {// 創建一次,多次復用let client = Client::new();let response1 = client.get("https://api.example.com/resource1").send()?;// 處理響應1...let response2 = client.get("https://api.example.com/resource2").send()?;// 處理響應2...Ok(())
}

2. 多線程處理(有限并發)

use reqwest::blocking::Client;
use std::thread;fn fetch_urls(urls: &[&str]) -> Vec<String> {let client = Client::new();let handles: Vec<_> = urls.iter().map(|url| {let url = (*url).to_string();thread::spawn(move || {match client.get(&url).send() {Ok(resp) => resp.text().unwrap_or_else(|_| "Error reading response".into()),Err(_) => "Request failed".into(),}})}).collect();handles.into_iter().map(|h| h.join().unwrap()).collect()
}fn main() {let urls = ["https://jsonplaceholder.typicode.com/posts/1","https://jsonplaceholder.typicode.com/posts/2","https://jsonplaceholder.typicode.com/posts/3",];let results = fetch_urls(&urls);for (i, result) in results.iter().enumerate() {println!("Response {}: {}", i + 1, result);}
}

📝 完整示例:天氣查詢工具

use reqwest::blocking::Client;
use serde::Deserialize;
use std::env;#[derive(Debug, Deserialize)]
struct WeatherData {name: String,main: Main,weather: Vec<Weather>,
}#[derive(Debug, Deserialize)]
struct Main {temp: f32,feels_like: f32,humidity: u32,
}#[derive(Debug, Deserialize)]
struct Weather {description: String,
}fn main() -> Result<(), Box<dyn std::error::Error>> {let api_key = env::var("OPENWEATHER_API_KEY").expect("請設置 OPENWEATHER_API_KEY 環境變量");let city = env::args().nth(1).unwrap_or_else(|| "London".to_string());let url = format!("https://api.openweathermap.org/data/2.5/weather?q={}&appid={}&units=metric",city, api_key);let client = Client::new();let response = client.get(&url).send()?;if !response.status().is_success() {return Err(format!("API請求失敗: {}", response.status()).into());}let weather: WeatherData = response.json()?;println!("\n{} 的天氣:", weather.name);println!("溫度: {:.1}°C", weather.main.temp);println!("體感溫度: {:.1}°C", weather.main.feels_like);println!("濕度: {}%", weather.main.humidity);println!("天氣狀況: {}", weather.weather[0].description);Ok(())
}

?? 同步方法的局限性

  1. 阻塞主線程:每個請求都會阻塞當前線程
  2. 低并發能力:不適合高并發場景
  3. 資源利用低效:線程在等待響應時無法處理其他任務
  4. 擴展性差:大規模請求需要大量線程

💡 何時使用同步方法

  1. 命令行工具:簡單的數據獲取工具
  2. 腳本任務:一次性數據處理腳本
  3. 低流量服務:內部工具或低流量API
  4. 學習階段:理解HTTP請求的基礎
  5. 簡單嵌入式系統:資源受限環境

📚 總結

同步 REST API 訪問核心步驟:

  1. 使用 reqwest::blocking 模塊
  2. 創建 Client 實例(可復用)
  3. 構建請求(GET/POST/PUT/DELETE)
  4. 發送請求并獲取響應
  5. 檢查狀態碼
  6. 解析響應體(文本/JSON/二進制)

最佳實踐:

  • 復用 Client:減少連接開銷
  • 設置超時:防止無限等待
  • 添加重試:處理臨時故障
  • 優雅錯誤處理:使用自定義錯誤類型
  • 環境變量管理:安全存儲API密鑰

對于需要高并發或高性能的場景,建議使用異步方法(如 reqwest 的異步API + tokio 運行時)。但對于許多應用場景,同步方法提供了簡單直接的解決方案。

本文來自互聯網用戶投稿,該文觀點僅代表作者本人,不代表本站立場。本站僅提供信息存儲空間服務,不擁有所有權,不承擔相關法律責任。
如若轉載,請注明出處:http://www.pswp.cn/diannao/94193.shtml
繁體地址,請注明出處:http://hk.pswp.cn/diannao/94193.shtml
英文地址,請注明出處:http://en.pswp.cn/diannao/94193.shtml

如若內容造成侵權/違法違規/事實不符,請聯系多彩編程網進行投訴反饋email:809451989@qq.com,一經查實,立即刪除!

相關文章

32.【.NET8 實戰--孢子記賬--從單體到微服務--轉向微服務】--單體轉微服務--財務服務--賬本與預算

在我們的孢子記賬應用中&#xff0c;賬本是用于記錄每一筆收支流水的核心模塊。通過賬本&#xff0c;我們可以清晰地追蹤資金的流入與流出&#xff0c;進行數據統計和分析&#xff0c;為后續的報表生成和決策支持提供基礎數據。預算模塊則是用于設置和管理預算的功能&#xff0…

模型預估打分對運籌跟蹤的影響

在uplift建模中&#xff0c;模型離線指標(QINI、AUUC)提升并不意味著在線A/B實驗的收益&#xff0c;因為在線運籌還需要λ\lambdaλ約束。如果模型打分不滿足單調增且roi邊際遞減&#xff0c;那么λ\lambdaλ運籌求解會非常不穩定&#xff0c;導致線上發券偏高&#xff0c;毛利…

音視頻學習(四十六):聲音的三要素

聲音是人類感知世界的重要途徑之一。在自然界中&#xff0c;聲波本質上是介質中傳播的機械振動&#xff0c;而人類對聲音的主觀感受主要通過三種屬性來認知和描述&#xff0c;即音調&#xff08;音高&#xff09;、響度&#xff08;強弱&#xff09;、音色&#xff08;音質&…

spring batch處理數據模板(Reader-Processor-Writer模式)

步驟監聽器 Component public class StepListener implements StepExecutionListener {private StepExecution stepExecution;public StepExecution getStepExecution() {return this.stepExecution;}Overridepublic void beforeStep(StepExecution stepExecution) {this.stepE…

【華為OD機試】從小桶里取球

題目描述 某部門開展Family Day開放日活動,其中有個從桶里取球的游戲,游戲規則如下: 有N個容量一樣的小桶等距排開,且每個小桶都默認裝了數不等的小球, 每個小桶裝的小球數量記錄在數組bucketBallNums中, 游戲開始時,要求所有桶的小球總數不能超過SUM, 如果小球總…

std::unordered_map 和 std::map的區別【C++】

std::unordered_map 和 std::map 是 C 標準庫中兩種不同的關聯容器&#xff0c;它們都用于存儲鍵值對&#xff0c;但在實現方式、性能特點和使用場景上存在顯著區別。以下是它們的主要區別&#xff1a; 1. 數據結構 std::map&#xff1a; 基于 紅黑樹&#xff08;一種自平衡二叉…

云原生環境里的顯示變革:Docker虛擬瀏覽器與cpolar穿透技術實戰

文章目錄前言【視頻教程】1. 關于neko2. 本地部署neko3. neko簡單使用4. 安裝內網穿透5. 配置neko公網地址6. 配置固定公網地址前言 現代遠程協作本該是無縫銜接的過程&#xff0c;卻被這些障礙不斷打斷&#xff1a;多設備屏幕同步存在延遲、跨平臺訪問需要復雜配置、公網IP申…

LVGL + ESP-Brookesia 在Windows下的編譯和運行

LVGL ESP-Brookesia 在Windows下的編譯和運行 1. 項目介紹 本項目是基于 LVGL&#xff08;輕量級多功能圖形庫&#xff09;和 ESP-Brookesia 的嵌入式模擬桌面應用開發框架&#xff0c;專為嵌入式設備構建豐富的圖形界面而設計。通過在Windows環境下模擬嵌入式設備的圖形界面…

【ip】IP地址能否直接填寫255?

IP地址數值限制? 最近有朋友后臺問我&#xff0c;IP地址里填255行不行&#xff1f;思索著有一陣子沒有分享基礎的知識&#xff0c;就在今天大致說一下&#xff0c;關于IP地址里填255行不行&#xff1f;答案當然是否定的。 IP地址由4個段組成&#xff0c;每個段的數值范圍其實限…

力扣熱題100----------141.環形鏈表

給你一個鏈表的頭節點 head &#xff0c;判斷鏈表中是否有環。 如果鏈表中有某個節點&#xff0c;可以通過連續跟蹤 next 指針再次到達&#xff0c;則鏈表中存在環。 為了表示給定鏈表中的環&#xff0c;評測系統內部使用整數 pos 來表示鏈表尾連接到鏈表中的位置&#xff08;索…

【Java開發日記】我們來說說 LockSupport 的 park 和 unpark

目錄 一、LockSupport 1.1、LockSupport函數列表 1.2、基本使用 先 park 再 unpark 先 unpark 再 park 1.3、特點 與 Object 的 wait & notify 相比 二、LockSupport park & unpark原理 2.1、情況一&#xff0c;先調用park&#xff0c;再調用unpark park 操作…

AGI|從“實驗室”到“生產線”:企業級AI Agent 如何突圍

在數字化轉型的深水區&#xff0c;企業級 AI Agent 正從技術概念走向產業實踐&#xff0c;成為驅動生產力變革的核心引擎。目錄 一、風口已至&#xff1a;AI Agent 的崛起邏輯與市場剛需 二、企業級AI Agent&#xff1a;核心能力與獨特價值定位 三、AI Agent 的未來目標 一、…

AtCoder Beginner Contest 417

文章目錄A A SubstringB Search and DeleteC Distance IndicatorsD Takahashis ExpectationE A Path in A DictionaryF Random GatheringG Binary CatAtCoder Beginner Contest 417A A Substring You are given an N-character string S consisting of lowercase English lett…

C++23 Concepts:用類型約束重構泛型編程的終極方案

一、開篇:模板元編程的"類型檢查困局" 某金融量化團隊曾遇到詭異bug: template<typename T> void process(T data) {static_assert(std::is_arithmetic<T>::value, "需要數值類型");// 業務邏輯... } 當調用process("hello")時…

【RK3568 看門狗驅動開發詳解】

RK3568 看門狗驅動開發詳解一、Linux 看門狗子系統架構?二、設備樹配置?三、 看門狗驅動實現四、驗證看門狗定時器&#xff08;Watchdog Timer&#xff09;是保障嵌入式系統可靠性的關鍵硬件&#xff0c;它通過定期接收 “喂狗” 信號監控系統運行狀態&#xff0c;當系統故障…

探索 Vue 3.6 新特性:Vapor Mode 與高性能 Web 應用開發

Vue 3.6 簡介 Vue.js 是一個廣受歡迎的漸進式 JavaScript 框架&#xff0c;以其簡潔的 API、靈活的組件系統和高性能著稱。Vue 3.6 是 Vue 3 系列的一個重要版本&#xff0c;引入了多項性能優化和新特性&#xff0c;尤其是備受關注的 Vapor Mode&#xff0c;這是一個無需虛擬 D…

初識prometheus

Prometheus&#xff1a;云原生時代的監控利器 在當今快速發展的云原生和微服務架構時代&#xff0c;傳統的監控系統面臨著巨大的挑戰&#xff1a;如何高效地收集海量、動態變化的指標&#xff1f;如何實時告警并快速定位問題&#xff1f;如何實現靈活的可視化和強大的數據查詢…

從源碼角度分析導致 JVM 內存泄露的 ThreadLocal

文章目錄1. 為什么需要ThreadLocal2. ThreadLocal的實現解析1.1 實現分析1.2 具體實現1.3 ThreadLocalMap中Hash沖突的解決1.3.1 Hash沖突解決的幾種方法1.3.1.1 開放定值法1.3.1.2 鏈地址法1.3.1.3再哈希法&#xff1a;1.3.1.4 建立公共溢出區1.3.2 ThreadLocal解決Hash沖突的…

React組件化的封裝

1. 組件化封裝的結構 1.1. 定義一個類(組件名必須是大寫&#xff0c;小寫會被認為是html元素), 繼續自React.Component1.2. 實現當前組件的render函數 render當中返回的jsx內容&#xff0c;就是之后React會幫助我們渲染的內容 1.3. 結構圖如下&#xff1a; data 方法render()…

嵌入式仿真教學的革新力量:深圳航天科技創新研究院引領高效學習新時代

嵌入式系統作為現代信息技術的核心基石&#xff0c;已深度融入工業控制、物聯網、智能終端等關鍵領域。高校肩負著培養嵌入式技術人才的重任&#xff0c;但傳統教學方式正面臨嚴峻挑戰&#xff1a;硬件實驗設備投入巨大、更新滯后、維護繁瑣、時空限制嚴格&#xff0c;難以滿足…