索引:https://waterflow.link/articles/1666534616841
咱们先看一个简略的例子,咱们自定义一个谬误,用来把多个谬误放在一起输入:
type CustomError struct { errors []string}func (c *CustomError) Add(err string) { c.errors = append(c.errors, err)}func (c *CustomError) Error() string { return strings.Join(c.errors, ";")}
因为实现了Error() string
办法,所以它实现了error接口。
当初咱们要实现一个增加课件的性能,然而增加之前须要验证参数的合法性,所以咱们创立了一个Validate办法,咱们可能会这么写:
package mainimport ( "errors" "fmt" "strings")type CustomError struct { errors []string}func (c *CustomError) Add(err error) { c.errors = append(c.errors, err.Error())}func (c *CustomError) Error() string { return strings.Join(c.errors, ";")}type Courseware struct { Name string Code string}func (c *Courseware) Validate() error { var m *CustomError // 1 if c.Name == "" { // 2 m = &CustomError{} m.Add(errors.New("课件名不能为空")) } if c.Code == "" { // 3 if m == nil { m = &CustomError{} } m.Add(errors.New("课件编号不能为空")) } return m // 4}func main() { m := Courseware{ Name: "多媒体课件", Code: "CW330", } if err := m.Validate(); err != nil { fmt.Println("valid err: ", err) }}
看上去如同一点问题都没有:
- 定义一个CustomError类型的指针
- 如果Name为空,初始化m,调用Add办法把谬误增加到CustomError.errors
- 如果Code为空,如果m还没有初始化,先初始化,调用Add办法把谬误增加到CustomError.errors
- 最初返回自定义谬误
然而当咱们执行下面的代码时,会发现后果并不是咱们想要的:
go run 8.govalid err: <nil>
咱们发现竟然走到了打印谬误的判断里,然而打印进去的谬误竟然是一个nil
。
在 Go 中,咱们必须晓得指针接收器能够为 nil。咱们看一个简略的例子:
package mainimport ( "fmt")type Demo struct {}func (d *Demo) Print() string { return "demo"}func main() { var d *Demo fmt.Println(d) fmt.Println(d.Print())}
go run 8.go<nil>demo
Demo被初始化为nil,然而这段代码能够失常运行。阐明nil指针也能够作为接收器。
其实下面的Print办法等价于:
func Print(d *Demo) string { return "demo"}
因为将 nil 指针传递给函数是无效的。 所以应用 nil 指针作为接收器也是无效的。
咱们持续回到下面的自定义谬误。
m 被初始化为指针的零值:nil。 如果所有验证都通过,return 语句返回的后果不是 nil,而是一个 nil 指针。 因为 nil 指针是一个无效的接收器,所以将后果转换为error接口不会产生 nil 值。
所以咱们尽管返回了一个nil指针,然而转换为error接口时并不是一个nil的接口(尽管是nil指针,然而是*CustomError类型,并实现了error)。
要解决这个问题,咱们只有间接返回nil值,不返回nil的指针:
package mainimport ( "errors" "fmt" "strings")type CustomError struct { errors []string}func (c *CustomError) Add(err error) { c.errors = append(c.errors, err.Error())}func (c *CustomError) Error() string { return strings.Join(c.errors, ";")}type Courseware struct { Name string Code string}func (c *Courseware) Validate() error { var m *CustomError if c.Name == "" { m = &CustomError{} m.Add(errors.New("课件名不能为空")) } if c.Code == "" { if m == nil { m = &CustomError{} } m.Add(errors.New("课件编号不能为空")) } // 这里如果m指针为nil,间接返回nil if m == nil { return nil } return m}func main() { m := Courseware{ Name: "多媒体课件", Code: "CW330", } if err := m.Validate(); err != nil { fmt.Println("valid err: ", err) }}
或者咱们间接返回*CustomError类型的谬误:
package mainimport ( "errors" "fmt" "strings")type CustomError struct { errors []string}func (c *CustomError) Add(err error) { c.errors = append(c.errors, err.Error())}func (c *CustomError) Error() string { return strings.Join(c.errors, ";")}type Courseware struct { Name string Code string}// 返回*CustomErrorfunc (c *Courseware) Validate() *CustomError { var m *CustomError if c.Name == "" { m = &CustomError{} m.Add(errors.New("课件名不能为空")) } if c.Code == "" { if m == nil { m = &CustomError{} } m.Add(errors.New("课件编号不能为空")) } return m}func main() { m := Courseware{ Name: "多媒体课件", Code: "CW330", } if err := m.Validate(); err != nil { fmt.Println("valid err: ", err) }}
但这并不是可取的,为了扩大咱们实现了error接口,也须要返回error类型的谬误。