搜尋
首頁後端開發Golang在sync.WaitGroup goroutine中寫入chan
在sync.WaitGroup goroutine中寫入chanFeb 09, 2024 pm 05:00 PM
同步機制

在sync.WaitGroup goroutine中写入chan

php小編柚子在這裡為大家介紹在sync.WaitGroup goroutine中寫入chan的方法。在並發程式設計中,sync.WaitGroup是一種非常有用的同步機制,它可以等待一組goroutine的執行完成。然而,有時我們需要在goroutine執行完畢後,將結果寫入到一個chan中,以供其他goroutine消費。本文將詳細介紹如何在sync.WaitGroup goroutine中實現這項功能,讓我們一起來看看吧!

問題內容

我正在從 API 端點取得項目清單。然後,對於每個項目,我都會發出另一個 API 請求以獲取有關單一項目的資料。

我無法同時對每個專案發出第二個 API 請求,因為我的 API 令牌有速率限制,如果我同時發出太多請求,我會受到限制。

但是,初始 API 回應資料可以分為多個頁面,這使我能夠同時處理資料頁面。

經過一些研究,下面的程式碼完全符合我的要求:

func main() {
    // pretend paginated results from initial API request
    page1 := []int{1, 2, 3}
    page2 := []int{4, 5, 6}
    page3 := []int{7, 8, 9}
    pages := [][]int{page1, page2, page3}

    results := make(chan string)

    var wg sync.WaitGroup
    for i := range pages {
        wg.Add(1)
        go func(i int) {
            defer wg.Done()
            for j := range pages[i] {
                // simulate making additional API request and building the report
                time.Sleep(500 * time.Millisecond)

                result := fmt.Sprintf("Finished creating report for %d", pages[i][j])
                results <- result
            }

        }(i)
    }

    go func() {
        wg.Wait()
        close(results)
    }()

    for result := range results {
        fmt.Println(result)
    }
}

我想了解為什麼它能發揮作用:

go func() {
    wg.Wait()
    close(results)
}()

我的第一次嘗試沒有成功——我想我可以在wg.Wait() 之後遍歷通道,並且我會在結果寫入results 通道時讀取結果。

func main() {
    // pretend paginated results from initial API request
    page1 := []int{1, 2, 3}
    page2 := []int{4, 5, 6}
    page3 := []int{7, 8, 9}
    pages := [][]int{page1, page2, page3}

    results := make(chan string)

    var wg sync.WaitGroup
    for i := range pages {
        wg.Add(1)
        go func(i int) {
            defer wg.Done()
            for j := range pages[i] {
                // simulate making additional API request and building the report
                time.Sleep(500 * time.Millisecond)

                result := fmt.Sprintf("Finished creating report for %d", pages[i][j])
                results <- result
            }

        }(i)
    }

    // does not work
    wg.Wait()
    close(results)

    for result := range results {
        fmt.Println(result)
    }
}

解決方法

在您的第一次嘗試:

  1. 主 goroutine 使 3 個 goroutine 將值放入結果通道中。
  2. 主協程等待所有協程完成。
  3. 其中一個 goroutine 將一個值放入結果通道中並填滿通道(通道大小為 1 個字串)。
  4. 現在所有三個 goroutine 都無法再將值放入結果通道並進入睡眠狀態,直到結果通道被釋放。
  5. 所有 goroutine 都處於睡眠狀態。你陷入了僵局。

在第二次嘗試:

  1. 主 goroutine 包含 4 個 goroutine。
  2. 3 個 goroutine 將值放入結果通道中。
  3. 其他 goroutine(我稱之為第 4 個)等待這 3 個 goroutine 結束。
  4. 同時主協程等待結果通道中的值(for 迴圈)
  5. 在這種情況下,如果其中一個 goroutine 在結果通道中放入一個值,則會阻塞其餘的三個 goroutine;主 Goroutine 將值從結果通道中取出,從而解除對其他 Goroutine 的阻塞。
  6. 因此,所有 3 個 goroutine 都放入各自的值並結束
  7. 然後第四個 goroutine 關閉通道
  8. 主 Goroutine 結束其 for 迴圈。

以上是在sync.WaitGroup goroutine中寫入chan的詳細內容。更多資訊請關注PHP中文網其他相關文章!

陳述
本文轉載於:stackoverflow。如有侵權,請聯絡admin@php.cn刪除
Linux环境编程必须搞懂的几个概念Linux环境编程必须搞懂的几个概念Feb 15, 2024 am 08:03 AM

对于初学者来说,要想在Linux环境下编程,必须深入理解一些重要概念才能更好地编写代码,实现业务功能。下面我们将介绍几个重要且常用的知识点。掌握这些概念可以避免在将来的编码中出现混淆。系统调用“❝所有操作系统的内核中都有一些内置函数,这些函数可以用来完成一些系统级别的功能。在Linux系统中,这些函数被称为“系统调用”(systemcall)。它们代表了从用户空间到内核空间的一种转换。❞”已收到消息.对于初学者来说,要想在Linux环境下编程,必须深入理解一些重要概念才能更好地编写代码,实现业务

Linux 进程间通信的方法和技巧:如何让进程之间互相交流和协作Linux 进程间通信的方法和技巧:如何让进程之间互相交流和协作Feb 11, 2024 pm 02:09 PM

进程间通信是指在Linux系统中,不同的进程之间进行数据的传递和共享,以实现进程之间的交流和协作。进程间通信的目的是提高系统的并发性和效率,以完成一些复杂的任务和功能。进程间通信的方法有很多种,如管道、消息队列、信号、共享内存、信号量、套接字等,它们各有各的特点和优缺点,适用于不同的场景和需求。但是,你真的了解Linux进程间通信的方法吗?你知道如何在Linux下使用和选择合适的进程间通信方法吗?你知道如何在Linux下优化和提高进程间通信的效果吗?本文将为你详细介绍Linux进程间通信的相关知

作为嵌入式开发工程师,关于Linux kernel同步机制你不得不知道作为嵌入式开发工程师,关于Linux kernel同步机制你不得不知道Feb 12, 2024 pm 02:50 PM

前言同步是进程之间,以及进程与系统资源之间的交互。由于Linux内核采用多任务,因此在多个进程之间必须有同步机制来保证协调。Linux内核中有许多种同步机制。今天我们将重点介绍kernel中的异步和同步机制,其中着重介绍kernel中的异步机制。kernel中的异步机制分为两种:一种是应用层的同步机制,即应用层线程之间的通信;另一种是内核的同步机制。当一个线程进入内核态后,它可以直接与内核沟通。kernel中有两个线程是这样的:一个是线程A,它进入内核态后会直接与内核沟通,告诉它要做什么,完成后

Golang中同步机制对于游戏开发性能的提升Golang中同步机制对于游戏开发性能的提升Sep 27, 2023 am 09:25 AM

Golang中同步机制对于游戏开发性能的提升,需要具体代码示例引言:游戏开发是一个对性能高要求的领域,在处理实时交互的同时,还要保持游戏的流畅性和稳定性。而Go语言(Golang)则提供了一种高效的编程语言和并发模型,使得其在游戏开发中有着广泛应用的潜力。本文将重点探讨Golang中同步机制对于游戏开发性能的提升,并通过具体代码示例来加深理解。一、Golan

详解Linux内核中的RCU机制详解Linux内核中的RCU机制Feb 10, 2024 pm 09:09 PM

Linux内核是一个复杂的系统,它需要处理多种多样的并发问题,如进程调度、内存管理、设备驱动、网络协议等。为了保证数据的一致性和正确性,Linux内核提供了多种同步机制,如自旋锁、信号量、读写锁等。但是,这些同步机制都有一些缺点,比如:自旋锁会导致CPU浪费时间在忙等待上,而且不能在抢占式内核中使用;信号量会导致进程睡眠和唤醒,增加了上下文切换的开销;读写锁会导致写者饥饿或者读者饥饿,而且在读者多写者少的情况下,写者还要获取锁的开销。那么,有没有一种更好的同步机制呢?答案是有的,那就是RCU(R

Linux多线程互斥量:一种保证线程安全的同步机制Linux多线程互斥量:一种保证线程安全的同步机制Feb 13, 2024 pm 01:40 PM

Linux系统是一种支持多任务并发执行的操作系统,它可以同时运行多个进程,从而提高系统的利用率和效率。但是,如果一个进程中有多个线程,而这些线程需要共享一些数据或资源,就可能出现数据不一致或资源竞争的问题,导致系统的错误或异常。为了解决这个问题,就需要使用一些同步机制,例如信号量、条件变量、互斥量等。其中,互斥量是一种比较简单而有效的同步机制,它可以让一个线程在访问共享数据或资源时,锁定它们,防止其他线程同时访问,从而保证线程安全。本文将介绍Linux系统中多线程互斥量的互斥的方法,包括互斥量的

了解Java中volatile的作用:保证多线程间数据的可见性和有序性了解Java中volatile的作用:保证多线程间数据的可见性和有序性Jan 30, 2024 am 08:53 AM

了解Java中volatile的作用:保证多线程间数据的可见性和有序性,需要具体代码示例在Java多线程编程中,为了确保多个线程之间的数据同步,我们常常需要使用volatile关键字。volatile关键字可以保证可见性和有序性,确保多个线程对某个变量的读写操作是正确的。一、可见性在多线程环境下,如果一个线程对某个共享变量进行修改,那么其他线程是否能够立即看

深入探讨Java多线程:同步和死锁的原理解析深入探讨Java多线程:同步和死锁的原理解析Feb 18, 2024 pm 08:01 PM

Java多线程原理剖析:线程同步与死锁问题分析摘要:本文将深入探讨Java多线程编程中的线程同步和死锁问题。通过详细解释线程的原理和Java提供的同步机制,我们将讨论如何正确地使用同步机制来避免线程冲突和数据不一致的问题。同时,我们还将分析死锁问题以及如何避免和解决这些问题。1.引言随着计算机硬件的发展,多核处理器已经成为现代计算机系统的标配。而多线程编程

See all articles

熱AI工具

Undresser.AI Undress

Undresser.AI Undress

人工智慧驅動的應用程序,用於創建逼真的裸體照片

AI Clothes Remover

AI Clothes Remover

用於從照片中去除衣服的線上人工智慧工具。

Undress AI Tool

Undress AI Tool

免費脫衣圖片

Clothoff.io

Clothoff.io

AI脫衣器

AI Hentai Generator

AI Hentai Generator

免費產生 AI 無盡。

熱門文章

R.E.P.O.能量晶體解釋及其做什麼(黃色晶體)
2 週前By尊渡假赌尊渡假赌尊渡假赌
倉庫:如何復興隊友
4 週前By尊渡假赌尊渡假赌尊渡假赌
Hello Kitty Island冒險:如何獲得巨型種子
4 週前By尊渡假赌尊渡假赌尊渡假赌

熱工具

Dreamweaver CS6

Dreamweaver CS6

視覺化網頁開發工具

SecLists

SecLists

SecLists是最終安全測試人員的伙伴。它是一個包含各種類型清單的集合,這些清單在安全評估過程中經常使用,而且都在一個地方。 SecLists透過方便地提供安全測試人員可能需要的所有列表,幫助提高安全測試的效率和生產力。清單類型包括使用者名稱、密碼、URL、模糊測試有效載荷、敏感資料模式、Web shell等等。測試人員只需將此儲存庫拉到新的測試機上,他就可以存取所需的每種類型的清單。

Safe Exam Browser

Safe Exam Browser

Safe Exam Browser是一個安全的瀏覽器環境,安全地進行線上考試。該軟體將任何電腦變成一個安全的工作站。它控制對任何實用工具的訪問,並防止學生使用未經授權的資源。

EditPlus 中文破解版

EditPlus 中文破解版

體積小,語法高亮,不支援程式碼提示功能

mPDF

mPDF

mPDF是一個PHP庫,可以從UTF-8編碼的HTML產生PDF檔案。原作者Ian Back編寫mPDF以從他的網站上「即時」輸出PDF文件,並處理不同的語言。與原始腳本如HTML2FPDF相比,它的速度較慢,並且在使用Unicode字體時產生的檔案較大,但支援CSS樣式等,並進行了大量增強。支援幾乎所有語言,包括RTL(阿拉伯語和希伯來語)和CJK(中日韓)。支援嵌套的區塊級元素(如P、DIV),