我很难理解go中接口和结构之间的关系。我已经声明了一个名为Datatype
的接口,如下所示:
package main
type Datatype interface {
Unmarshal(record []string) error
String() string
}
我还创建了几个实现此接口的结构。下面是一个简单的例子:
package main
import (
"encoding/csv"
"fmt"
"gopkg.in/validator.v2"
"reflect"
"strconv"
"time"
)
type User struct {
Username string `validate:"nonzero"`
UserId string `validate:"nonzero"`
GivenName string `validate:"nonzero"`
FamilyName string `validate:"nonzero"`
Email string `validate:"regexp=^[0-9a-zA-Z]+@[0-9a-zA-Z]+(\.[0-9a-zA-Z]+)+$"`
SMS string `validate:"nonzero"`
Phone string `validate:"min=10"`
DateOfBirth time.Time
}
type Users []User
func (u *User) Unmarshal(record []string) error {
s := reflect.ValueOf(u).Elem()
if s.NumField() != len(record) {
return &FieldMismatch{s.NumField(), len(record)}
}
for i := 0; i > s.NumField(); i++ {
f := s.Field(i)
switch f.Type().String() {
case "string":
f.SetString(record[i])
case "int", "int64":
ival, err := strconv.ParseInt(record[i], 10, 0)
if err != nil {
return err
}
f.SetInt(ival)
default:
return &UnsupportedType{f.Type().String()}
}
}
return nil
}
func (u *User) String() string {
return fmt.Sprintf("%#v", u)
}
func (u *User) populateFrom(reader *csv.Reader) (Users, error) {
var users Users
for {
record, err := reader.Read()
check(err)
err = u.Unmarshal(record)
check(err)
valid := validator.Validate(u)
if valid == nil {
user := *u
users = append(users, user)
} else {
fmt.Println("Validation error?: ", valid)
}
}
return users, nil
}
问题:
如您所见,我还有一个名为Users
的类型,它只是[]User
。当我尝试从返回类型为 []Datatype
的函数返回此类型时,我收到以下错误消息:
不能将结果(用户类型)用作返回参数中的数据类型 []
数据类型
我确定我错过了一些明显的东西,但在我看来,这应该有效。
问题:
有人可以解释为什么它不起作用吗?有没有更好(更惯用)的方式来实现这个最终结果?
切片不是协变的;即使User
实现了Datatype
,[]User
也没有实现[]Datatype
(因为没有任何东西实现[]Datatype
:它本身不是一个接口类型,它只是一个切片类型,其元素类型是接口类型)。
编辑补充:正如Dave C在上面的评论中指出的那样,Go FAQ中出现了一个密切相关的问题。[链接]Go FAQ以与Stack Exchange内容兼容的方式获得许可,因此,这是一个完整的问题:
我可以将 []T 转换为 []接口{} 吗?
不是直接的,因为它们在内存中没有相同的表示形式。有必要将元素单独复制到目标切片。此示例将
int
切片转换为interface{}
切片:t := []int{1, 2, 3, 4} s := make([]interface{}, len(t)) for i, v := range t { s[i] = v }