我正在尝试创建一个以mysql rows.Scan函数需要的方式转换结构的函数,因此我不需要手动传递大量参数.
注意:我知道sqlx的存在以及在每个指针的单独行中手动编写的替代方法,但是我想以这种方式解决它,因为我正在学习并想要了解正在发生的事情.
我用这个解决方案得到的错误是:
panic: sql: Scan error on column index 0: destination not a pointer
对我来说看起来valueField.Addr().Pointer()应该是一个指向价值的指针.以下是我的代码的简化.
type User struct {
Name string
Age int
}
func StrutForScan(u interface{}) []interface{} {
val := reflect.ValueOf(u).Elem()
v := make([]interface{}, val.NumField())
for i := 0; i < val.NumField(); i++ {
valueField := val.Field(i)
v[i] = valueField.Addr().Pointer()
}
return v
}
func ListUsers {
rows, err := db.Query("SELECT * FROM users")
PanicIf(err)
var user User
for rows.Next() {
err := rows.Scan(StrutForScan(&user)...)
PanicIf(err)
fmt.Printf("\nName: %s, Age: %s", user.Name, string(user.Age))
}
}
Run Code Online (Sandbox Code Playgroud)
One*_*One 11
您需要使用.Interface()不.Pointer()
func StrutForScan(u interface{}) []interface{} {
val := reflect.ValueOf(u).Elem()
v := make([]interface{}, val.NumField())
for i := 0; i < val.NumField(); i++ {
valueField := val.Field(i)
v[i] = valueField.Addr().Interface()
}
return v
}
Run Code Online (Sandbox Code Playgroud)
这背后的原因是.Pointer()返回数据的实际"指针",如果不使用unsafe包,就不能对它做很多事情.