这取决于您用来与数据库进行对话的内容,但是如果您正在使用
database/sql并且提供了对此提供支持的驱动程序,则可以让您的
Contract类型实现Valuer接口。
type Service struct { ID string `json:"id,omitempty" db:"id"` Name string `json:"name" db:"name"` Contract Contract `json:"contract" db:"contract"`}type Contract struct { ServiceTime int `json:"service_time"` Region string `json:"region"`}func (c *Contract) Value() (driver.Value, error) { if c != nil { b, err := json.Marshal(c) if err != nil { return nil, err } return string(b), nil } return nil, nil}然后,当您执行查询时,只需在
Service.Contract字段中传递,驱动程序应调用该
Value方法。
sql := // INSERT INTO ...svc := // &Service{ ...db.Exec(sql, svc.ID, svc.Name, svc.Contract)为了能够从db中检索blob并将其解组,
Contract您可以再次实现Scanner接口,如果类型实现了该目标,则驱动程序应该调用它。
func (c *Contract) Scan(src interface{}) error { var data []byte if b, ok := src.([]byte); ok { data = b } else if s, ok := src.(string); ok { data = []byte(s) } return json.Unmarshal(data, c)}// ...sql := // SELECt * FROM ...svc := // &Service{ ...db.QueryRow(sql, 123).Scan(&svc.ID, &svc.Name, &svc.Contract)


