【问题标题】:Why my go coroutines is stuck after processing?为什么我的 go coroutines 处理后卡住了?
【发布时间】:2019-02-01 22:59:37
【问题描述】:

我是 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
}

【问题讨论】:

标签: loops go channel goroutine go-gorm


【解决方案1】:

从没有人准备发送值块的无缓冲通道(例如您的通道)接收。这就是你所经历的。 Spec: Receive operator:

表达式 [&lt;-c] 阻塞,直到有值可用。

在通道情况下发出“EOF”信号的常用方法是在没有更多值要发送时从发送方关闭通道,使用内置的close() 函数。

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

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")
}

【讨论】:

  • 感谢您的回答。也可以告诉我,如果这段代码可以进一步改进?喜欢使用缓冲通道?
  • @MayukhSarkar 使用缓冲通道不会简化代码。如果生成或使用值可能需要大量时间,这很有用,因此如果一个参与者比另一个参与者更快/更慢,他们也不会减慢彼此的速度。
  • 哦,这里的生产者是数据库,消费者是 csv 编写器,所以我想我不需要任何缓冲通道。我可以通过其他方式改进代码的执行吗?
猜你喜欢
  • 2016-11-01
  • 1970-01-01
  • 1970-01-01
  • 1970-01-01
  • 1970-01-01
  • 1970-01-01
  • 1970-01-01
  • 2016-03-09
  • 1970-01-01
相关资源
最近更新 更多