AVt天堂网 手机版,亚洲va久久久噜噜噜久久4399,天天综合亚洲色在线精品,亚洲一级Av无码毛片久久精品

當前位置:首頁 > 科技  > 軟件

異步Rust:構建實時消息代理服務器

來源: 責編: 時間:2024-02-01 12:44:22 229觀看
導讀在本文中,我們將深入研究使用Rust構建實時消息代理服務器,展示其強大的并發特性。我們將使用Warp作為web服務器,并使用Tokio來管理異步任務。此外,我們將創建一個WebSocket客戶端來測試代理服務器的功能。設計圖如下:圖片

在本文中,我們將深入研究使用Rust構建實時消息代理服務器,展示其強大的并發特性。我們將使用Warp作為web服務器,并使用Tokio來管理異步任務。此外,我們將創建一個WebSocket客戶端來測試代理服務器的功能。nEj28資訊網——每日最新資訊28at.com

設計圖如下:nEj28資訊網——每日最新資訊28at.com

圖片圖片nEj28資訊網——每日最新資訊28at.com

nEj28資訊網——每日最新資訊28at.com

構建消息代理服務器

消息代理服務器允許客戶端為主題生成事件并訂閱它們。它使用Warp作為HTTP和WebSocket服務器,使用Tokio作為異步運行時。nEj28資訊網——每日最新資訊28at.com

使用以下命令創建一個Rust項目:nEj28資訊網——每日最新資訊28at.com

cargo new real-ime-message

在Cargo.toml文件中加入以下依賴項:nEj28資訊網——每日最新資訊28at.com

[dependencies]futures-util = "0.3.30"tokio = {version = "1.35.1", features = ["full"]}tokio-tungstenite = "0.21.0"url = "2.5.0"warp = "0.3.6"

在src/main.rs文件中定義一個Broker結構體:nEj28資訊網——每日最新資訊28at.com

use std::{    collections::{HashMap, VecDeque},    sync::Arc,};use futures_util::{SinkExt, StreamExt};use tokio::sync::{    mpsc::{self, UnboundedSender},    RwLock,};use warp::{filters::ws::Message, Filter};type Topic = String;type Event = String;type WsSender = UnboundedSender<warp::ws::Message>;struct Broker {    events: Arc<RwLock<HashMap<Topic, VecDeque<Event>>>>,    subscribers: Arc<RwLock<HashMap<Topic, Vec<WsSender>>>>,}
  • events:存儲每個主題的事件。
  • subscribers:跟蹤每個主題的訂閱者。

創建一個新的Broker實例:nEj28資訊網——每日最新資訊28at.com

impl Broker {    fn new() -> Self {        Broker {            events: Arc::new(RwLock::new(HashMap::new())),            subscribers: Arc::new(RwLock::new(HashMap::new())),        }    }}

定義發布事件的方法produce:nEj28資訊網——每日最新資訊28at.com

impl Broker {    ......    async fn produce(&self, topic: Topic, event: Event) {        let mut events = self.events.write().await;        events            .entry(topic.clone())            .or_default()            .push_back(event.clone());        // 異步通知所有訂閱者        let subscribers_list;        {            let subscribers = self.subscribers.read().await;            subscribers_list = subscribers.get(&topic).cloned().unwrap_or_default();        }        for ws_sender in subscribers_list {            // 將事件發送到WebSocket客戶端            let _ = ws_sender.send(warp::ws::Message::text(event.clone()));        }    }}

nEj28資訊網——每日最新資訊28at.com

這個方法主要是將事件添加到相應的主題,然后將新事件通知所有訂閱者。nEj28資訊網——每日最新資訊28at.com

定義subscribe方法,來管理新的訂閱:nEj28資訊網——每日最新資訊28at.com

impl Broker {    ......    pub async fn subscribe(&self, topic: Topic, socket: warp::ws::WebSocket) {        let (ws_sender, mut ws_receiver) = socket.split();        let (tx, mut rx) = mpsc::unbounded_channel::<Message>();        {            let mut subs = self.subscribers.write().await;            subs.entry(topic).or_default().push(tx);        }        tokio::task::spawn(async move {            while let Some(result) = ws_receiver.next().await {                match result {                    Ok(message) => {                        // 處理有效的消息                        if message.is_text() {                            println!(                                "Received message from client: {}",                                message.to_str().unwrap()                            );                        }                    }                    Err(e) => {                        // 處理錯誤                        eprintln!("WebSocket error: {:?}", e);                        break;                    }                }            }            println!("WebSocket connection closed");        });        tokio::task::spawn(async move {            let mut sender = ws_sender;            while let Some(msg) = rx.recv().await {                let _ = sender.send(msg).await;            }        });    }}

nEj28資訊網——每日最新資訊28at.com

這個方法主要是將WebSocket拆分為發送方和接收方,將訂閱者添加到訂閱者列表中,處理傳入的WebSocket消息。nEj28資訊網——每日最新資訊28at.com

main函數代碼如下:nEj28資訊網——每日最新資訊28at.com

#[tokio::main]async fn main() {    let broker = Arc::new(Broker::new());    let broker_clone1 = Arc::clone(&broker);    let broker_clone2 = Arc::clone(&broker);    let produce = warp::path!("produce" / String)        .and(warp::post())        .and(warp::body::json())        .and(warp::any().map(move || Arc::clone(&broker_clone1)))        .and_then(            move |topic: String, event: Event, broker_clone2: Arc<Broker>| async move {                broker_clone2.produce(topic, event).await;                Ok::<_, warp::Rejection>(warp::reply())            },        );    let subscribe = warp::path!("subscribe" / String).and(warp::ws()).map(        move |topic: String, ws: warp::ws::Ws| {            let broker_clone3 = Arc::clone(&broker_clone2);            ws.on_upgrade(move |socket| async move {                broker_clone3.subscribe(topic.clone(), socket).await;            })        },    );    let routes = produce.or(subscribe);    println!("Broker server running at http://127.0.0.1:3030");    warp::serve(routes).run(([127, 0, 0, 1], 3030)).await;}

nEj28資訊網——每日最新資訊28at.com

nEj28資訊網——每日最新資訊28at.com

實現WebSocket客戶端

WebSocket客戶端將模擬一個訂閱主題和接收消息的真實用戶。nEj28資訊網——每日最新資訊28at.com

在src/bin目錄下,創建一個ws_cli.rs文件。在文件中定義websocket_client函數,建立WebSocket連接并管理消息:nEj28資訊網——每日最新資訊28at.com

use futures_util::{sink::SinkExt, stream::StreamExt};use std::sync::Arc;use tokio::sync::RwLock;use tokio::time::{sleep, Duration};use tokio_tungstenite::{connect_async, tungstenite::protocol::Message};use url::Url;async fn websocket_client(topic_url: &str) {    // 解析要連接WebSocket服務器的URL    let url = Url::parse(topic_url).expect("Invalid URL");    // 連接到WebSocket服務器    let (ws_stream, _) = connect_async(url).await.expect("Failed to connect");    println!("WebSocket client connected");    let (mut write, mut read) = ws_stream.split();    let message = Arc::new(RwLock::new(String::new()));    let message_1 = message.clone();    // 生成一個任務來處理傳入的消息    tokio::spawn(async move {        let msg_lock = message_1.clone();        while let Some(message) = read.next().await {            match message {                Ok(msg) => {                    let mut ms = msg_lock.write().await;                    *ms = msg.to_text().unwrap().to_string();                    println!("Received message: {}", msg.to_text().unwrap());                }                Err(e) => {                    eprintln!("Error receiving message: {:?}", e);                    break;                }            }        }    });    // 發送消息    loop {        let msg_lock = message.clone();        let ms = msg_lock.read().await;        if let Err(e) = write.send(Message::Text(ms.to_string())).await {            eprintln!("Error sending message: {:?}", e);            break;        }        sleep(Duration::from_secs(5)).await;    }}

nEj28資訊網——每日最新資訊28at.com

main函數代碼如下:nEj28資訊網——每日最新資訊28at.com

#[tokio::main]async fn main() {    websocket_client("ws://127.0.0.1:3030/subscribe/newtopic").await;}

nEj28資訊網——每日最新資訊28at.com

nEj28資訊網——每日最新資訊28at.com

測試

執行如下命令運行消息代理服務器:nEj28資訊網——每日最新資訊28at.com

cargo run --bin real-ime-message

nEj28資訊網——每日最新資訊28at.com

執行結果:nEj28資訊網——每日最新資訊28at.com

Broker server running at http://127.0.0.1:3030

nEj28資訊網——每日最新資訊28at.com

nEj28資訊網——每日最新資訊28at.com

然后打開一個新的命令行,執行如下命令運行WebSocket客戶端:nEj28資訊網——每日最新資訊28at.com

cargo run --bin ws_cli

nEj28資訊網——每日最新資訊28at.com

執行結果:nEj28資訊網——每日最新資訊28at.com

WebSocket client connected

nEj28資訊網——每日最新資訊28at.com

nEj28資訊網——每日最新資訊28at.com

向http://127.0.0.1:3030/produce/newtopic接口發送post請求,如圖:nEj28資訊網——每日最新資訊28at.com

圖片圖片nEj28資訊網——每日最新資訊28at.com

nEj28資訊網——每日最新資訊28at.com

客戶端接收到消息:nEj28資訊網——每日最新資訊28at.com

WebSocket client connectedReceived message: This is a new event

nEj28資訊網——每日最新資訊28at.com

nEj28資訊網——每日最新資訊28at.com

nEj28資訊網——每日最新資訊28at.com

總結

我們已經探索了在Rust中創建一個簡單的消息代理,并使用WebSocket客戶端對其進行測試。這個例子突出了Rust在構建高效、并發的網絡應用程序方面的能力。nEj28資訊網——每日最新資訊28at.com

nEj28資訊網——每日最新資訊28at.com

本文鏈接:http://www.tebozhan.com/showinfo-26-70392-0.html異步Rust:構建實時消息代理服務器

聲明:本網頁內容旨在傳播知識,若有侵權等問題請及時與本網聯系,我們將在第一時間刪除處理。郵件:2376512515@qq.com

上一篇: 分享 15 個 HTML 新特性,大多數人可能不知道,建議盡早使用上

下一篇: PHP 高性能的事件循環庫 Revolt

標簽:
  • 熱門焦點
  • 十個可以手動編寫的 JavaScript 數組 API

    JavaScript 中有很多API,使用得當,會很方便,省力不少。 你知道它的原理嗎? 今天這篇文章,我們將對它們進行一次小總結。現在開始吧。1.forEach()forEach()用于遍歷數組接收一參
  • SpringBoot中使用Cache提升接口性能詳解

    環境:springboot2.3.12.RELEASE + JSR107 + Ehcache + JPASpring 框架從 3.1 開始,對 Spring 應用程序提供了透明式添加緩存的支持。和事務支持一樣,抽象緩存允許一致地使用各
  • .NET 程序的 GDI 句柄泄露的再反思

    一、背景1. 講故事上個月我寫過一篇 如何洞察 C# 程序的 GDI 句柄泄露 文章,當時用的是 GDIView + WinDbg 把問題搞定,前者用來定位泄露資源,后者用來定位泄露代碼,后面有朋友反
  • 虛擬鍵盤 API 的妙用

    你是否在遇到過這樣的問題:移動設備上有一個固定元素,當激活虛擬鍵盤時,該元素被隱藏在了鍵盤下方?多年來,這一直是 Web 上的默認行為,在本文中,我們將探討這個問題、為什么會發生
  • JVM優化:實戰OutOfMemoryError異常

    一、Java堆溢出堆內存中主要存放對象、數組等,只要不斷地創建這些對象,并且保證 GC Roots 到對象之間有可達路徑來避免垃 圾收集回收機制清除這些對象,當這些對象所占空間超過
  • 郭明錤稱華為和江淮汽車合作開發問界MPV,定價100萬左右、計劃明年量產

    8 月 1 日消息,郭明錤今天在 Medium 平臺發布博文,稱華為正在和江淮汽車合作,開發售價在 100 萬元的問界 MPV,預計在 2024 年第 2 季度量產,銷量目標為
  • 三星Galaxy Z Fold5今日亮相:厚度縮減但仍略顯厚重

    據官方此前宣布,三星將于7月26日也就是今天在韓國首爾舉辦Unpacked活動,屆時將帶來帶來包括Galaxy Buds 3、Galaxy Watch 6、Galaxy Tab S9、Galaxy
  • OPPO K11樣張首曝:千元機影像“卷”得真不錯!

    一直以來,OPPO K系列機型都保持著較為均衡的產品體驗,歷來都是2K價位的明星機型,去年推出的OPPO K10和OPPO K10 Pro兩款機型憑借各自的出色配置,堪稱有
  • 由于成本持續增加,筆記本產品價格預計將明顯上漲

    根據知情人士透露,由于材料、物流等成本持續增加,筆記本產品價格預計將在2021年下半年有明顯上漲。進入6月下旬以來,全球半導體芯片缺貨情況加劇,顯卡、處理器
Top