【发布时间】:2019-04-11 23:03:51
【问题描述】:
我是 Golang 的新手,我发现 Go 频道非常有趣。我的背景来自 JavaScript,我想在 Go 中按顺序处理并发请求,有点像 JavaScript 中的Promise.all()。我想要的只是发出一些并发运行的请求,并按照我调用它们的顺序处理返回的数据。
等效的 JavaScript 代码如下:
async function main() {
// Assuming all db calls will return a promise
const firstUserPromise = firstDbCall().then((res) => res);
const secondUserPromise = secondDbCall().then((res) => res);
const thridUserPromise = thridDbCall().then((res) => res);
const [
firstUserData,
secondUserData,
thirdUserData
] = await Promise.all([firstUserPromise, secondUserPromise, thirdUserPromise]);
}
如果您不熟悉 JavaScript,在上面的代码中,我会同时进行三个数据库调用(从第 3 行到第 5 行)。然后我在等待他们给出一些回应(从第 7 行到第 10 行)。这段代码的好处在于,当所有三个数据库调用都完成后,我将按照等待它们的顺序获得结果。 firstUserData 会收到来自firstUserPromise 的回复,secondUserData 会收到来自secondUserPromise 等等。
下面是一个假设的代码,我希望它等同于上面的 JavaScript 代码:
package main
import "fmt"
func main() {
set = make(chan, string)
get = make(chan, string)
// First DB call
go firstDbCall()
// Second DB call
go secondDbCall()
// Third DB call
go thirdDbCall()
// How to make sending data to channels predictable
// First data to `set` channel will send data to firstDbCall
// Second one will `set` to secondDbCall and so on.
set <- "userId 1"
set <- "userId 2"
set <- "userId 3"
// Similarly, How to make receiving data from channels predictable
// firstUserData will data of "userId 1", secondUserData will have
// data of "userId 2" and so on.
firstUserData := <-get
secondUserData := <-get
thirdUserData := <-get
}
由于从通道获取数据是不可预测的,我怎样才能让它们像 JavaScript 代码一样可预测?
【问题讨论】:
-
让每个 goroutine 将其结果填充到切片中自己的元素中。
-
@Mohammad_Aziz 关注第一条评论。它不仅说明您的问题的重复性,而且还提及您正在寻找的答案。如果这没有帮助,请在需要时回复此社区的帮助。
-
为什么不将函数作为参数传递给 db 调用,并在那里处理?
-
您关心函数的执行顺序还是关心结果处理的顺序?如果是第一个,你应该按顺序做事,没有渠道或套路。如果是第二个,则使用 WaitGroup 等待所有 goroutine 完成,让它们将结果存储在预定义的切片或数组中,并以您认为正确的任何顺序处理结果数组。
-
我不知道
WaitGroup以及它是如何工作的。我确实关心处理结果的顺序@MadWombat。
标签: javascript go concurrency promise channel