首页 > 文章列表 > 使用database/sql包中的Scanner接口来实现数据读取功能

使用database/sql包中的Scanner接口来实现数据读取功能

418 2024-02-02
问题内容

如何实现database/sql.Scanner接口?

在此查询中,SELECT 子句中有 3 个字段:

  • idsmallint 无符号
  • is_suspended tinyint 无符号
  • name varchar

database/sql中,3列的数据类型是:

  • int64
  • int64
  • []uint8

这适用于 []接口{},但希望将每个列类型直接实现到 database/sql.Scanner 接口

cols    := make([]interface{}, 3)
ptr     := make([]interface{}, 3)

for i, _ := range cols {
    ptr[i] = &cols[i]
}

if err := rows.Scan(ptr...); err != nil {
    fmt.Println("err:", err)
}

// pair column data with column name
res := map[string]any
for i, name := range res_cols {
    res[name] = *ptr[i].(*any)
    
    fmt.Printf("Type: %T %sn", res[name], name)
}

我尝试做的事情,但无法真正使其发挥作用

类型

type Type_int int

func (t *Type_int) Scan(value interface{}) error {
    switch value := value.(type) {
    case int64:
        *t = Type_int(value)
    default:
        return fmt.Errorf("Invalid database type: %T %v", value, value)
    }
    return nil
}

type Type_string string

func (t *Type_string) Scan(value interface{}) error {
    switch value := value.(type) {
    case []uint8:
        *t = Type_string(value)
    default:
        return fmt.Errorf("Invalid database type: %T %v", value, value)
    }
    return nil
}

代码

ptr     := make([]interface{}, 3)

cols    := []interface{}{
    Type_int,
    Type_int,
    Type_string,
}

for i, _ := range cols {
    ptr[i] = &cols[i]
}

if err := rows.Scan(ptr...); err != nil {
    fmt.Println("err:", err)
}

// pair column data with column name
res := map[string]any
for i, name := range res_cols {
    res[name] = *ptr[i].(*any)
    
    fmt.Printf("Type: %T %sn", res[name], name)
}


正确答案


使用指向给定类型值的指针初始化 ptr

var c1 Type_int
var c2 Type_int
var c3 Type_string
ptr := []any{&c1, &c2, &c3}
if err := rows.Scan(ptr...); err != nil {
    fmt.Println("err:", err)
}

ptr 切片在上面的代码片段中没有提供任何值。代码可以简化为:

var c1 Type_int
var c2 Type_int
var c3 Type_string
if err := rows.Scan(&c1, &c2, &c3); err != nil {
    fmt.Println("err:", err)
}