gpt4 book ai didi

loops - 为什么我的 go 协程在处理后卡住了?

转载 作者:IT王子 更新时间:2023-10-29 01:51:56 24 4
gpt4 key购买 nike

我是 Golang 的新手。我一直在使用 GORM 和 go 并发来读取 SQLite 数据库并将其写入 CSV 文件。它工作顺利但是当处理完成时它并没有结束主程序并退出。我必须打印 command+c 才能退出。我不知道我做错了什么。可能是它进入了某种阻塞或死锁模式之类的。此外,它也不打印再见消息。这意味着它仍在尝试从 channel 中读取数据。请帮忙。这是代码。

package main

import (
"fmt"
"reflect"

"github.com/jinzhu/gorm"
_ "github.com/jinzhu/gorm/dialects/sqlite"
)

type AirQuality struct {
// gorm.Model
// ID uint `gorm:"column:id"`
Index string `gorm:"column:index"`
BEN string `gorm:"column:BEN"`
CH4 string `gorm:"column:CH4"`
CO string `gorm:"column:CO"`
EBE string `gorm:"column:EBE"`
MXY string `gorm:"column:MXY"`
NMHC string `gorm:"column:NMHC"`
NO string `gorm:"column:NO"`
NO2 string `gorm:"column:NO_2"`
NOX string `gorm:"column:NOx"`
OXY string `gorm:"column:OXY"`
O3 string `gorm:"column:O_3"`
PM10 string `gorm:"column:PM10"`
PM25 string `gorm:"column:PM25"`
PXY string `gorm:"column:PXY"`
SO2 string `gorm:"column:SO_2"`
TCH string `gorm:"column:TCH"`
TOL string `gorm:"column:TOL"`
Time string `gorm:"column:date; type:timestamp"`
Station string `gorm:"column:station"`
}

func (AirQuality) TableName() string {
return "AQ"
}

func main() {
c := generateRows("boring!!")
for {
fmt.Println(<-c)
if c == nil {
fmt.Println("Bye")
break
}
}
}

func generateRows(msg string) <-chan []string {
c := make(chan []string)
go func() {
db, err := gorm.Open("sqlite3", "./load_testing_7.6m.db")
if err != nil {
panic("failed to connect database")
}
defer db.Close()
rows, err := db.Model(&AirQuality{}).Limit(20).Rows()
defer rows.Close()
if err != nil {
panic(err)
}
for rows.Next() {
var aq AirQuality
db.ScanRows(rows, &aq)
v := reflect.Indirect(reflect.ValueOf(aq))
var buf []string
for i := 0; i < v.NumField(); i++ {
buf = append(buf, v.Field(i).String())
}
c <- buf
}
}()
return c
}

最佳答案

从没有人准备好发送值 block 的无缓冲 channel (例如您的 channel )接收。这就是你的体验。 Spec: Receive operator:

The expression [<-c] blocks until a value is available.

在 channel 情况下发出“EOF”信号的常用方法是在没有更多值可发送时从发送方关闭 channel ,使用内置 close() 功能。

尝试从关闭的 channel 接收可以立即继续,产生 zero value channel 的元素类型。要检测此“关闭”状态,请使用特殊的逗号-ok 习惯用法:

value, ok := <- c

如果 channel 关闭,ok将是 false (否则为 true )。

“排空” channel 直到它关闭的简单而正确的方法是使用 for range循环,像这样:

for value := range c {
fmt.Println("Received:", value)
}

for range一旦从 channel c 接收到所有值就终止在它关闭之前发送的。

所以里面generateRows() ,这样做:

go func() {
// // Use defer so it will be closed no matter how this function call ends
defer close(c)
// ...
}()

还有你的main() :

func main() {
c := generateRows("boring!!")
for v := range c {
fmt.Println(v)
}
fmt.Println("Bye")
}

关于loops - 为什么我的 go 协程在处理后卡住了?,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/52038739/

24 4 0
Copyright 2021 - 2024 cfsdn All Rights Reserved 蜀ICP备2022000587号
广告合作:1813099741@qq.com 6ren.com