日韩成人免费在线_国产成人一二_精品国产免费人成电影在线观..._日本一区二区三区久久久久久久久不

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

我們一起聊聊如何編寫異步運行時通用庫?

來源: 責編: 時間:2024-05-07 09:15:18 178觀看
導讀如果你正在用Rust編寫異步應用程序,在某些情況下,你可能希望將代碼分成幾個子crate。這樣做的好處是:更好的封裝,在子系統之間有一個crate邊界,可以產生更清晰的代碼和定義更良好的API。不再需要這樣寫:pub(crate)。更快的

如果你正在用Rust編寫異步應用程序,在某些情況下,你可能希望將代碼分成幾個子crate。這樣做的好處是:RhO28資訊網——每日最新資訊28at.com

  • 更好的封裝,在子系統之間有一個crate邊界,可以產生更清晰的代碼和定義更良好的API。不再需要這樣寫:pub(crate)。
  • 更快的編譯,通過將一個大crate分解成幾個獨立的小crate,它們可以并發地編譯。

使用一個異步運行時,編寫異步運行時通用庫的好處是什么?RhO28資訊網——每日最新資訊28at.com

  • 可移植性,你可以很容易地切換到不同的異步運行時或wasm。
  • 保證正確性,針對tokio和async-std,測試一個庫就可以發現更多的bug,包括并發bug(由于任務執行順序模糊)和“未定義行為”(由于誤解異步運行時實現細節)

下面使用三種方法來實現異步運行時通用庫。RhO28資訊網——每日最新資訊28at.com

方法1,定義自己的異步運行時TraitRhO28資訊網——每日最新資訊28at.com

使用futures crate,可以編寫非常通用的庫代碼,但是time,sleep或timeout等操作必須依賴于異步運行時。這時,你可以定義自己的AsyncRuntime trait,并要求下游實現它。RhO28資訊網——每日最新資訊28at.com

use std::{future::Future, time::Duration};pub trait AsyncRuntime: Send + Sync + 'static {    type Delay: Future<Output = ()> + Send;    // 返回值必須是一個Future    fn sleep(duration: Duration) -> Self::Delay;}

可以像這樣使用上面的庫代碼:RhO28資訊網——每日最新資訊28at.com

async fn operation<R: AsyncRuntime>() {    R::sleep(Duration::from_millis(1)).await;}

下面是它如何實現的:RhO28資訊網——每日最新資訊28at.com

pub struct TokioRuntime;impl AsyncRuntime for TokioRuntime {    type Delay = tokio::time::Sleep;    fn sleep(duration: Duration) -> Self::Delay {        tokio::time::sleep(duration)    }}#[tokio::main]async fn main() {    operation::<TokioRuntime>().await;    println!("Hello, world!");}

方法2,在內部抽象異步運行時并公開特性標志RhO28資訊網——每日最新資訊28at.com

為了處理網絡連接或文件句柄,我們可以使用AsyncRead / AsyncWrite trait:RhO28資訊網——每日最新資訊28at.com

#[async_trait]pub(crate) trait AsyncRuntime: Send + Sync + 'static {    type Connection: AsyncRead + AsyncWrite + Send + Sync + 'static;    async fn connect(addr: SocketAddr) -> std::io::Result<Self::Connection>;}

可以像這樣使用上面的庫代碼:RhO28資訊網——每日最新資訊28at.com

async fn operation<R: AsyncRuntime>(conn: &mut R::Connection) where    R::Connection: Unpin,{    conn.write(b"some bytes").await;}

然后為每個異步運行時定義一個模塊:RhO28資訊網——每日最新資訊28at.com

#[cfg(feature = "runtime-async-std")]mod async_std_impl;#[cfg(feature = "runtime-async-std")]use async_std_impl::*;#[cfg(feature = "runtime-tokio")]mod tokio_impl;#[cfg(feature = "runtime-tokio")]use tokio_impl::*;

tokio_impl模塊:RhO28資訊網——每日最新資訊28at.com

mod tokio_impl {    use std::net::SocketAddr;    use async_trait::async_trait;    use crate::AsyncRuntime;    pub struct TokioRuntime;    #[async_trait]    impl AsyncRuntime for TokioRuntime {        type Connection = tokio::net::TcpStream;        async fn connect(addr: SocketAddr) -> std::io::Result<Self::Connection> {            tokio::net::TcpStream::connect(addr).await        }    }}

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

#[tokio::main]async fn main() {    let mut conn =        TokioRuntime::connect(SocketAddr::new(IpAddr::from_str("0.0.0.0").unwrap(), 8080))            .await            .unwrap();    operation::<TokioRuntime>(&mut conn).await;    println!("Hello, world!");}

方法3,維護一個異步運行時抽象庫RhO28資訊網——每日最新資訊28at.com

基本上,將使用的所有異步運行時api寫成一個包裝器庫。這樣做可能很繁瑣,但也有一個好處,即可以在一個地方為項目指定與異步運行時的所有交互,這對于調試或跟蹤非常方便。RhO28資訊網——每日最新資訊28at.com

例如,我們定義異步運行時抽象庫的名字為:common-async-runtime,它的異步任務處理代碼如下:RhO28資訊網——每日最新資訊28at.com

// common-async-runtime/tokio_task.rspub use tokio::task::{JoinHandle as TaskHandle};pub fn spawn_task<F, T>(future: F) -> TaskHandle<T>where    F: Future<Output = T> + Send + 'static,    T: Send + 'static,{    tokio::task::spawn(future)}

async-std的任務API與Tokio略有不同,這需要一些樣板文件:RhO28資訊網——每日最新資訊28at.com

// common-async-runtime/async_std_task.rspub struct TaskHandle<T>(async_std::task::JoinHandle<T>);pub fn spawn_task<F, T>(future: F) -> TaskHandle<T>where    F: Future<Output = T> + Send + 'static,    T: Send + 'static,{    TaskHandle(async_std::task::spawn(future))}#[derive(Debug)]pub struct JoinError;impl std::error::Error for JoinError {}impl<T> Future for TaskHandle<T> {    type Output = Result<T, JoinError>;    fn poll(        mut self: std::pin::Pin<&mut Self>,        cx: &mut std::task::Context<'_>,    ) -> std::task::Poll<Self::Output> {        match self.0.poll_unpin(cx) {            std::task::Poll::Ready(res) => std::task::Poll::Ready(Ok(res)),            std::task::Poll::Pending => std::task::Poll::Pending,        }    }}

在Cargo.toml中,你可以簡單地將common-async-runtime作為依賴項包含進來。這使得你的庫代碼很“純粹”,因為現在選擇異步運行時是由下游控制的。與方法1類似,這個crate可以在沒有任何異步運行時的情況下編譯,這很簡潔!RhO28資訊網——每日最新資訊28at.com

本文鏈接:http://www.www897cc.com/showinfo-26-87039-0.html我們一起聊聊如何編寫異步運行時通用庫?

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

上一篇: Python 推導式在接口自動化里的運用

下一篇: .NET中的多線程超時處理實踐

標簽:
  • 熱門焦點
  • 一篇文章帶你了解 CSS 屬性選擇器

    屬性選擇器對帶有指定屬性的 HTML 元素設置樣式。可以為擁有指定屬性的 HTML 元素設置樣式,而不僅限于 class 和 id 屬性。一、了解屬性選擇器CSS屬性選擇器提供了一種簡單而
  • 使用LLM插件從命令行訪問Llama 2

    最近的一個大新聞是Meta AI推出了新的開源授權的大型語言模型Llama 2。這是一項非常重要的進展:Llama 2可免費用于研究和商業用途。(幾小時前,swyy發現它已從LLaMA 2更名為Lla
  • 三分鐘白話RocketMQ系列—— 如何發送消息

    我們知道RocketMQ主要分為消息 生產、存儲(消息堆積)、消費 三大塊領域。那接下來,我們白話一下,RocketMQ是如何發送消息的,揭秘消息生產全過程。注意,如果白話中不小心提到相關代
  • 共享單車的故事講到哪了?

    來源丨海克財經與共享充電寶相差不多,共享單車已很久沒有被國內熱點新聞關照到了。除了一再漲價和用戶直呼用不起了。近日多家媒體再發報道稱,成都、天津、鄭州等地多個共享單
  • ESG的面子與里子

    來源 | 光子星球撰文 | 吳坤諺編輯 | 吳先之三伏大幕拉起,各地高溫預警不絕,但處于厄爾尼諾大&ldquo;烤&rdquo;之下的除了眾生,還有各大企業發布的ESG報告。ESG是&ldquo;環境保
  • OPPO、vivo、小米等國內廠商Q2在印度智能手機市場份額依舊高達55%

    7月20日消息,據外媒報道,研究機構的報告顯示,在全球智能手機出貨量同比仍在下滑的大背景下,印度這一有潛力的市場也未能幸免,出貨量同比也有下滑,多家廠
  • 華為發布HarmonyOS 4:更好玩、更流暢、更安全

    在8月4日的華為開發者大會2023(HDC.Together)大會上,HarmonyOS 4正式發布。自2019年發布以來,HarmonyOS一直以用戶為中心,經歷四年多的發展HarmonyOS已
  • 三星顯示已開始為AR設備研發硅基LED微顯示屏

    7月18日消息,據外媒報道,隨著蘋果首款頭顯產品Vision Pro在6月份正式推出,AR/VR/MR等頭顯產品也就將成為各大公司下一個重要的競爭領域,對顯示屏這一關
  • 由于成本持續增加,筆記本產品價格預計將明顯上漲

    根據知情人士透露,由于材料、物流等成本持續增加,筆記本產品價格預計將在2021年下半年有明顯上漲。進入6月下旬以來,全球半導體芯片缺貨情況加劇,顯卡、處理器
Top 主站蜘蛛池模板: 巴林左旗| 长治市| 青河县| 涞源县| 肃北| 永胜县| 华安县| 灵宝市| 马鞍山市| 乐东| 武城县| 甘德县| 九江县| 祁东县| 聂拉木县| 平度市| 邻水| 盐边县| 新宾| 临汾市| 浙江省| 漯河市| 兴山县| 武穴市| 乐清市| 松原市| 屯留县| 库尔勒市| 蒙自县| 怀集县| 阿巴嘎旗| 汉阴县| 义马市| 江川县| 泗洪县| 屯昌县| 新竹市| 抚顺市| 老河口市| 客服| 施秉县|