簡體   English   中英

為什么加工后的協程卡住了?

[英]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.

 
粵ICP備18138465號  © 2020-2024 STACKOOM.COM