![](/img/trans.png)
[英]Why does my primality test stop after 1 and not seem to be able to go on?
[英]Why my go coroutines is stuck after processing?
我是Golang的新手。 我一直在使用GORM和go的並發性來讀取SQLite數據庫並將其寫入CSV文件。 它工作順利,但是在完成處理后,並沒有結束主程序並退出。 我必須打印command+c
退出。 我不知道我在做什么錯。 可能是正在進入某種阻塞或死鎖模式之類的東西。 而且它也不是打印再見消息。 這意味着它仍在嘗試從通道讀取數據。 請幫忙。 這是代碼。
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
}
從沒有人准備發送值塊的無緩沖通道(例如您的通道)接收。 這就是您的經驗。 規格:接收運算符:
表達式[
<-c
]阻塞,直到有值可用為止。
在通道情況下發出“ EOF”信號的常用方法是,當沒有更多值要發送時,使用內置的close()
函數從發送方關閉該通道。
嘗試從封閉的通道接收可能會立即進行,從而產生通道元素類型的零值 。 要檢測此“關閉”狀態,請使用特殊的逗號習慣用法:
value, ok := <- c
如果關閉了通道,則ok
將為false
(否則為true
)。
一種簡單而正確的“排水”通道直到關閉通道的方法是使用for range
循環,如下所示:
for value := range c {
fmt.Println("Received:", value)
}
一旦在關閉通道之前已從通道c
接收到所有值,則for range
終止。
因此,在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")
}
聲明:本站的技術帖子網頁,遵循CC BY-SA 4.0協議,如果您需要轉載,請注明本站網址或者原文地址。任何問題請咨詢:yoyou2525@163.com.