带缓冲通道的循环
For loop with buffered channel
我正在尝试使用 Go 频道,但遇到以下简单程序未终止的问题。
基本上我想发出一些异步 HTTP get 请求,然后 wait 直到它们 all 完成。我正在使用缓冲通道,但我不确定这是惯用的方式。
func GetPrice(quotes chan string) {
client := &http.Client{}
req, _ := http.NewRequest("GET", "https://some/api", nil)
req.Header.Set("Accept", "application/json")
res, err := client.Do(req)
if err != nil {
panic(err)
}
defer res.Body.Close()
body, err := ioutil.ReadAll(res.Body)
quotes <- string(body)
}
func main() {
const max = 3
quotes := make(chan string, max)
for i := 0; i < max; i++ {
go GetPrice(quotes)
}
for n := range quotes {
fmt.Printf("\n%s", n)
}
}
程序成功打印 3(最多)个项目
{"price":"1.00"}
{"price":"2.00"}
{"price":"3.00"}
但随后阻塞并且永远不会退出。
另一种方法:
package main
import (
"fmt"
"io/ioutil"
"net/http"
)
func GetPrice(quotes chan string) {
client := &http.Client{}
req, _ := http.NewRequest("GET", "https://some/api", nil)
req.Header.Set("Accept", "application/json")
res, err := client.Do(req)
if err != nil {
panic(err)
}
defer res.Body.Close()
body, err := ioutil.ReadAll(res.Body)
quotes <- string(body)
}
func run(quotes chan string, quit chan bool, max int) {
for num := range quotes {
fmt.Println(num)
max--
if max == 0 {
quit <- true
}
}
}
func main() {
const max = 3
quotes := make(chan string)
quit := make(chan bool)
go run(quotes, quit, max)
for i := 0; i < max; i++ {
go GetPrice(quotes)
}
<-quit
}
sync.WaitGroup
可以在这里使用等待所有goroutines然后关闭quotes
通道:
func getPrice(quotes chan<- string, onExit func()) {
go func() {
defer onExit()
req, _ := http.NewRequest("GET", "https://some/api", nil)
req.Header.Set("Accept", "application/json")
client := &http.Client{}
res, err := client.Do(req)
if err != nil {
panic(err) // should be handled properly
}
defer res.Body.Close()
body, err := ioutil.ReadAll(res.Body)
quotes <- string(body)
}()
}
func main() {
const max = 3
var wg sync.WaitGroup
quotes := make(chan string, max)
for i := 0; i < max; i++ {
wg.Add(1)
getPrice(quotes, func() { wg.Done() })
}
go func() {
defer close(quotes)
wg.Wait()
}()
for n := range quotes {
fmt.Printf("\n%s", n)
}
}
func GetPrice(quotes chan string) {
client := &http.Client{}
req, _ := http.NewRequest("GET", "https://some/api", nil)
req.Header.Set("Accept", "application/json")
res, err := client.Do(req)
if err != nil {
panic(err)
}
defer res.Body.Close()
body, err := ioutil.ReadAll(res.Body)
quotes <- string(body)
}
func Consumer(chan string){
for n ,ok:= range quotes {
if !ok {
break
}
fmt.Printf("\n%s", n)
}
}
func main() {
const max = 3
quotes := make(chan string, max)
for i := 0; i < max; i++ {
go GetPrice(quotes)
}
go Consumer(quotes)
close(quotes)
}
我正在尝试使用 Go 频道,但遇到以下简单程序未终止的问题。
基本上我想发出一些异步 HTTP get 请求,然后 wait 直到它们 all 完成。我正在使用缓冲通道,但我不确定这是惯用的方式。
func GetPrice(quotes chan string) {
client := &http.Client{}
req, _ := http.NewRequest("GET", "https://some/api", nil)
req.Header.Set("Accept", "application/json")
res, err := client.Do(req)
if err != nil {
panic(err)
}
defer res.Body.Close()
body, err := ioutil.ReadAll(res.Body)
quotes <- string(body)
}
func main() {
const max = 3
quotes := make(chan string, max)
for i := 0; i < max; i++ {
go GetPrice(quotes)
}
for n := range quotes {
fmt.Printf("\n%s", n)
}
}
程序成功打印 3(最多)个项目
{"price":"1.00"}
{"price":"2.00"}
{"price":"3.00"}
但随后阻塞并且永远不会退出。
另一种方法:
package main
import (
"fmt"
"io/ioutil"
"net/http"
)
func GetPrice(quotes chan string) {
client := &http.Client{}
req, _ := http.NewRequest("GET", "https://some/api", nil)
req.Header.Set("Accept", "application/json")
res, err := client.Do(req)
if err != nil {
panic(err)
}
defer res.Body.Close()
body, err := ioutil.ReadAll(res.Body)
quotes <- string(body)
}
func run(quotes chan string, quit chan bool, max int) {
for num := range quotes {
fmt.Println(num)
max--
if max == 0 {
quit <- true
}
}
}
func main() {
const max = 3
quotes := make(chan string)
quit := make(chan bool)
go run(quotes, quit, max)
for i := 0; i < max; i++ {
go GetPrice(quotes)
}
<-quit
}
sync.WaitGroup
可以在这里使用等待所有goroutines然后关闭quotes
通道:
func getPrice(quotes chan<- string, onExit func()) {
go func() {
defer onExit()
req, _ := http.NewRequest("GET", "https://some/api", nil)
req.Header.Set("Accept", "application/json")
client := &http.Client{}
res, err := client.Do(req)
if err != nil {
panic(err) // should be handled properly
}
defer res.Body.Close()
body, err := ioutil.ReadAll(res.Body)
quotes <- string(body)
}()
}
func main() {
const max = 3
var wg sync.WaitGroup
quotes := make(chan string, max)
for i := 0; i < max; i++ {
wg.Add(1)
getPrice(quotes, func() { wg.Done() })
}
go func() {
defer close(quotes)
wg.Wait()
}()
for n := range quotes {
fmt.Printf("\n%s", n)
}
}
func GetPrice(quotes chan string) {
client := &http.Client{}
req, _ := http.NewRequest("GET", "https://some/api", nil)
req.Header.Set("Accept", "application/json")
res, err := client.Do(req)
if err != nil {
panic(err)
}
defer res.Body.Close()
body, err := ioutil.ReadAll(res.Body)
quotes <- string(body)
}
func Consumer(chan string){
for n ,ok:= range quotes {
if !ok {
break
}
fmt.Printf("\n%s", n)
}
}
func main() {
const max = 3
quotes := make(chan string, max)
for i := 0; i < max; i++ {
go GetPrice(quotes)
}
go Consumer(quotes)
close(quotes)
}