成人免费xxxxx在线视频软件_久久精品久久久_亚洲国产精品久久久_天天色天天色_亚洲人成一区_欧美一级欧美三级在线观看

我們一起聊聊如何編寫異步運(yùn)行時(shí)通用庫?

開發(fā) 前端
在Cargo.toml中,你可以簡(jiǎn)單地將common-async-runtime作為依賴項(xiàng)包含進(jìn)來。這使得你的庫代碼很“純粹”,因?yàn)楝F(xiàn)在選擇異步運(yùn)行時(shí)是由下游控制的。與方法1類似,這個(gè)crate可以在沒有任何異步運(yùn)行時(shí)的情況下編譯,這很簡(jiǎn)潔!?

如果你正在用Rust編寫異步應(yīng)用程序,在某些情況下,你可能希望將代碼分成幾個(gè)子crate。這樣做的好處是:

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

使用一個(gè)異步運(yùn)行時(shí),編寫異步運(yùn)行時(shí)通用庫的好處是什么?

  • 可移植性,你可以很容易地切換到不同的異步運(yùn)行時(shí)或wasm。
  • 保證正確性,針對(duì)tokio和async-std,測(cè)試一個(gè)庫就可以發(fā)現(xiàn)更多的bug,包括并發(fā)bug(由于任務(wù)執(zhí)行順序模糊)和“未定義行為”(由于誤解異步運(yùn)行時(shí)實(shí)現(xiàn)細(xì)節(jié))

下面使用三種方法來實(shí)現(xiàn)異步運(yùn)行時(shí)通用庫。

方法1,定義自己的異步運(yùn)行時(shí)Trait

使用futures crate,可以編寫非常通用的庫代碼,但是time,sleep或timeout等操作必須依賴于異步運(yùn)行時(shí)。這時(shí),你可以定義自己的AsyncRuntime trait,并要求下游實(shí)現(xiàn)它。

use std::{future::Future, time::Duration};

pub trait AsyncRuntime: Send + Sync + 'static {
    type Delay: Future<Output = ()> + Send;

    // 返回值必須是一個(gè)Future
    fn sleep(duration: Duration) -> Self::Delay;
}

可以像這樣使用上面的庫代碼:

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

下面是它如何實(shí)現(xiàn)的:

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,在內(nèi)部抽象異步運(yùn)行時(shí)并公開特性標(biāo)志

為了處理網(wǎng)絡(luò)連接或文件句柄,我們可以使用AsyncRead / AsyncWrite trait:

#[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>;
}

可以像這樣使用上面的庫代碼:

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

然后為每個(gè)異步運(yùn)行時(shí)定義一個(gè)模塊:

#[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模塊:

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函數(shù)代碼:

#[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,維護(hù)一個(gè)異步運(yùn)行時(shí)抽象庫

基本上,將使用的所有異步運(yùn)行時(shí)api寫成一個(gè)包裝器庫。這樣做可能很繁瑣,但也有一個(gè)好處,即可以在一個(gè)地方為項(xiàng)目指定與異步運(yùn)行時(shí)的所有交互,這對(duì)于調(diào)試或跟蹤非常方便。

例如,我們定義異步運(yùn)行時(shí)抽象庫的名字為:common-async-runtime,它的異步任務(wù)處理代碼如下:

// common-async-runtime/tokio_task.rs

pub 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的任務(wù)API與Tokio略有不同,這需要一些樣板文件:

// common-async-runtime/async_std_task.rs

pub 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中,你可以簡(jiǎn)單地將common-async-runtime作為依賴項(xiàng)包含進(jìn)來。這使得你的庫代碼很“純粹”,因?yàn)楝F(xiàn)在選擇異步運(yùn)行時(shí)是由下游控制的。與方法1類似,這個(gè)crate可以在沒有任何異步運(yùn)行時(shí)的情況下編譯,這很簡(jiǎn)潔!

責(zé)任編輯:武曉燕 來源: coding到燈火闌珊
相關(guān)推薦

2024-09-09 00:00:00

編寫技術(shù)文檔

2025-01-09 07:54:03

2023-11-29 07:10:50

python協(xié)程異步編程

2021-08-27 07:06:10

IOJava抽象

2024-02-20 21:34:16

循環(huán)GolangGo

2023-08-10 08:28:46

網(wǎng)絡(luò)編程通信

2023-08-04 08:20:56

DockerfileDocker工具

2023-06-30 08:18:51

敏捷開發(fā)模式

2023-09-10 21:42:31

2022-05-24 08:21:16

數(shù)據(jù)安全API

2023-04-03 00:09:13

2024-09-30 09:33:31

2024-11-27 16:07:45

2023-07-04 08:06:40

數(shù)據(jù)庫容器公有云

2022-12-05 09:10:21

2024-07-03 08:36:14

序列化算法設(shè)計(jì)模式

2022-11-12 12:33:38

CSS預(yù)處理器Sass

2025-03-27 02:00:00

SPIJava接口

2024-02-26 00:00:00

Go性能工具

2023-12-28 09:55:08

隊(duì)列數(shù)據(jù)結(jié)構(gòu)存儲(chǔ)
點(diǎn)贊
收藏

51CTO技術(shù)棧公眾號(hào)

主站蜘蛛池模板: 成人免费看电影 | 国产a视频 | 国产一级免费在线观看 | 欧美最猛黑人xxxx黑人 | 丁香五月网久久综合 | 久久99精品久久久久久琪琪 | 午夜免费视频 | 国产成人免费视频 | 久久亚洲精品视频 | 日韩精品视频在线观看一区二区三区 | 羞羞视频网站在线观看 | 国产精品久久久久影院色老大 | 久久久精品一区二区三区 | 免费特黄视频 | 在线观看av免费 | 国外成人在线视频 | 91精品国产一区二区三区 | 亚洲精品粉嫩美女一区 | 亚洲av毛片 | 北条麻妃99精品青青久久主播 | 亚洲精品国产第一综合99久久 | 欧美精品首页 | 国产中文视频 | 午夜精品久久久 | 精品国产乱码久久久久久a丨 | 国产激情精品视频 | 亚洲成人免费视频在线观看 | 日韩快播电影 | 免费国产一区二区 | 久久久成人一区二区免费影院 | 日韩中文字幕 | 午夜激情一区 | 成人精品免费 | 羞羞视频在线观看 | 国产精品中文字幕在线 | 日韩av成人 | 中文字幕一区二区三区日韩精品 | 亚洲精品99999 | 国产一区二区在线播放 | 欧美视频三区 | 久久精品久久精品 |