代码之家  ›  专栏  ›  技术社区  ›  laminatefish

Golang Mysql scan返回零,什么时候有数据?

  •  1
  • laminatefish  · 技术社区  · 6 年前

    有点挠头的时刻。我有以下结构:

    type Room struct {
        ID              int
        Name            string
        RoomType        int
        CreatedAt       time.Time
        UpdatedAt       time.Time
        DeletedAt       time.Time
        GroupId         int
        BlockId         int
        ProjectId       int
        RoomLength      float64
        RoomWidth       float64
        CeilingHeight   float64
        CeilingColorHex string
        WallColorHex    string
        FloorColorHex   string
        CeilingColorRgb string
        WallColorRgb    string
        FloorColorRgb   string
    }
    

    在rest api中读取:

    database := db.New()
    stmt, err := database.Prepare("SELECT * FROM room WHERE block_id = ?")
    if err != nil {
        panic(err)
    }
    defer stmt.Close()
    
    rows, err := stmt.Query(c.Param("id"))
    if err != nil {
        panic(err)
    }
    defer rows.Close()
    
    var rooms []common.Room
    for rows.Next() {
        var r common.Room
        err := rows.Scan(&r.ID, &r.Name, &r.RoomType, &r.CreatedAt, &r.UpdatedAt, &r.DeletedAt,
            &r.GroupId, &r.BlockId, &r.ProjectId, &r.RoomLength, &r.RoomWidth, &r.CeilingHeight,
            &r.CeilingColorHex, &r.WallColorHex, &r.FloorColorHex, &r.CeilingColorRgb, &r.WallColorRgb,
            &r.FloorColorRgb)
        if err = rows.Err(); err != nil {
            panic(err)
        }
    
        fmt.Printf("Found: %v", r)
        rooms = append(rooms, r)
    }
    

    然而,产生的有效载荷是:

    {3 Loki #1 3 2018-09-25 08:42:38 +0000 UTC 2018-09-25 14:52:39 +0000 UTC 0001-01-01 00:00:00 +0000 UTC 0 0 0 0 0 0      }
    

    我特别关注长度/宽度(上面的)是0。但在db中:

    mysql> select * from room where id = 3 \G
    *************************** 1. row ***************************
                   id: 3
                 name: Loki #1
            room_type: 3
           created_at: 2018-09-25 08:42:38
           updated_at: 2018-09-25 14:52:39
           deleted_at: NULL
             group_id: 0
             block_id: 1
           project_id: 0
          room_length: 10
           room_width: 7
       ceiling_height: 4
    ceiling_color_hex: #c0c0c0
       wall_color_hex: #a9a9a9
      floor_color_hex: #708090
    ceiling_color_rgb: 192,192,192
       wall_color_rgb: 169,169,169
      floor_color_rgb: 112,128,144
    1 row in set (0.00 sec)
    

    我认为这可能与不同的类型有关,但是在db中更改了它们,然后在代码中更改了它们之后,就没有更改了。有人能解释为什么吗 .Scan 是不是在获取某些值?

    谢谢!

    1 回复  |  直到 6 年前
        1
  •  6
  •   xiaomi    6 年前

    首先,检查来自 rows.Scan() . 您只能从 rows.Err()

    err := rows.Scan(&r.ID, &r.Name, &r.RoomType, &r.CreatedAt, &r.UpdatedAt, &r.DeletedAt,
        &r.GroupId, &r.BlockId, &r.ProjectId, &r.RoomLength, &r.RoomWidth, &r.CeilingHeight,
        &r.CeilingColorHex, &r.WallColorHex, &r.FloorColorHex, &r.CeilingColorRgb, &r.WallColorRgb,
        &r.FloorColorRgb)
    if err != nil {
        panic(err) // Error related to the scan
    }
    if err = rows.Err(); err != nil {
        panic(err) // Error related to the iteration of rows
    }
    

    当值从 deleted_at 返回为 NULL ,扫描将返回一个错误,例如 unsupported Scan, storing driver.Value type <nil> into type *time.Time 你的结构的其他部分会 零值 .

    这意味着您的房间结构必须更改以使用指向时间的指针。

    CreatedAt       *time.Time
    UpdatedAt       *time.Time
    DeletedAt       *time.Time
    

    parameter parseTime sql.Open("mysql", "user:password@/dbname?parseTime=true) 正确解析mysql的时间。