V2EX = way to explore
V2EX 是一个关于分享和探索的地方
现在注册
已注册用户请  登录
V2EX 提问指南
DollarKiller
V2EX  ›  问与答

请问 RUST 如何控制并发数量?

  •  
  •   DollarKiller · 2020-12-09 14:02:48 +08:00 · 717 次点击
    这是一个创建于 1488 天前的主题,其中的信息可能已经有所发展或是发生改变。

    golang 可以通过 channl 来实现并发数量的限制 如下

    	// slice 存储当前需要处理的任务
    	task := []string{
    		"task1",
    		"task2",
    		"task3",
    	}
    
    	limit := make(chan bool,10) // 定义空间为 10 的 chan 作为限流器
    
    	for i := range task {
    		idx := i  // copy
    		
    		limit<-true  // 每次任务执行  塞入
    		go func() {
    			defer func() {
    				<-limit  // 任务执行完毕 塞出
    			}()
    			// 处理任务
    			fmt.Println("task: ",task[idx])
    		}()
    	}
    

    rust 有什么好的解决方案呢? 我是这样写的

    use async_std::task;
    use std::time::Duration;
    use async_std::channel;
    
    #[async_std::main]
    async fn main() {
        let mut task_list = Vec::new();
        for i in 0..10000 {
            task_list.push(format!("task: {}",i))
        }
    
        let (sen,rec) = channel::bounded(10);
    
        for i in task_list {
            sen.send(1).await;
            let rec = rec.clone();
            task::spawn(async move {
                task::sleep(Duration::from_millis(500)).await;
                println!("Processing {}",i);
                rec.recv().await;
            });
        }
    
        loop {
            task::sleep(Duration::from_secs(1)).await;
        }
    }
    

    如果使用上面方式解决 可能回出现处理 task 时发生异常以至于没有执行 rec.recv().await

    go 通过 defer 确定 task 处理完毕时必然回执行 处理令牌 , rust 怎么解决呢?

    使用一个结构体处理 task 并重写它的 Deref trait ?

    1 条回复    2021-09-13 18:50:02 +08:00
    kxuanobj
        1
    kxuanobj  
       2021-09-13 18:50:02 +08:00   ❤️ 1
    控制并发量可以用 https://docs.rs/futures/0.3.16/futures/stream/trait.StreamExt.html#method.buffer_unordered

    实现 defer 可以这样:

    ```rust
    fn run() {

    struct Defer {}
    impl Drop for Defer {
    fn drop(&mut self) {
    println!("defer")
    }
    }

    let f = Defer {};
    println!("what ever");

    // cancel defer
    //std::mem::forget(f);
    }
    ```
    关于   ·   帮助文档   ·   博客   ·   API   ·   FAQ   ·   实用小工具   ·   3001 人在线   最高记录 6679   ·     Select Language
    创意工作者们的社区
    World is powered by solitude
    VERSION: 3.9.8.5 · 28ms · UTC 13:48 · PVG 21:48 · LAX 05:48 · JFK 08:48
    Developed with CodeLauncher
    ♥ Do have faith in what you're doing.