我需要从数据库中读取日期,将其转换为某个时间戳并将其转换为 JSON。

我有以下代码:

package usages

import (
    "fmt"
    "time"
)

type SpecialOffer struct {
    PublishedDate   jsonTime    `gorm:"column:publishing_date" json:"published_date"`
    ExpirationDate  jsonTime    `gorm:"column:expiration_date" json:"expiration_date"`
}

type jsonTime struct {
    time.Time
}

func (tt jsonTime) MarshalJSON() ([]byte, error) {
    jsonTime := fmt.Sprintf("\"%s\"", tt.Format("20060102"))
    return []byte(jsonTime), nil
}

当我像这样运行它时,出现以下错误:
sql: Scan error on column index 8, name "publishing_date": unsupported Scan, storing driver.Value type time.Time into type *usages.trvTime

而且数据是错误的:
{"published_date":"00010101","expiration_date":"00010101"}

如果我将 SpecialOffer 结构更改为使用 time.Time ,它返回正确,但显然格式错误:
{"published_date":"2020-03-12T00:00:00Z","expiration_date":"2020-06-12T00:00:00Z"}

我究竟做错了什么?

最佳答案

您应该实现 sql.Scanner driver.Valuer 接口(interface)。

像这样的东西:

func (j *jsonTime) Scan(src interface{}) error {
    if t, ok := src.(time.Time); ok {
        j.Time = t
    }
    return nil
}

func (j jsonTime) Value() (driver.Value, error) {
    return j.Time, nil
}

这是必要的,因为 database/sql 和其他一些 go ORM(如果不是全部)使用的 gorm 包仅提供对少数类型的开箱即用支持。

大多数支持的类型是语言的基本内置类型,如 stringintbool 等。通过扩展,它还支持任何自定义用户定义类型,其底层类型是上述基本类型之一,然后支持 []byte 类型和相关的 sql.RawBytes 类型,最后 time.Time 类型也支持 ootb。

您可能想要写入或读取数据库的任何其他类型都需要实现上述两个接口(interface)。 sql.ScannerScan 方法在列的值被解码为支持的类型之一后自动调用(这就是为什么您需要针对 time.Time 而不是针对 []byte 键入断言的原因)。 driver.ValuerValue 方法在驱动程序将其编码为对目标列有效的格式之前自动调用(这就是为什么您可以直接返回 time.Time 而不是自己进行编码)。

请记住
type jsonTime struct {
    time.Time
}

甚至
type jsonTime time.Time

声明了一个不等于 time.Time 的新类型,这就是为什么 database/sql 包没有选择它的原因。

关于go - 将时间从数据库转换为自定义时间失败,我们在Stack Overflow上找到一个类似的问题:https://stackoverflow.com/questions/61630216/

10-16 06:37