nil指针被发送给chan,但收到"非零"

Kev*_*uan 9 null go

package main

import (
    "fmt"
    "os/exec"
)

func main() {
    errChan := make(chan error)
    go func() {
        var e *exec.Error = nil
        errChan <- e
    }()
    err := <-errChan
    if err != nil {
        fmt.Printf("err != nil, but err = %v\n", err)
    }
}
Run Code Online (Sandbox Code Playgroud)

输出很奇怪:err != nil, but err = <nil> 在这里试试:http://play.golang.org/p/_iyh0m7O1a

ANi*_*sus 13

问题在于作为错误接口传递到通道的值不是nil,而是exec.Error指向nil 的指针.

如果您更改,程序将正常运行:

go func() {
    var e *exec.Error = nil
    if e == nil {
        errChan <- nil
    }
}()
Run Code Online (Sandbox Code Playgroud)

这是解决问题的适当方法,因为报告没有发生错误的惯用方法是传递一个nil错误接口.

但是,如果你想改变main(可能是因为你使用了第三方软件包而错误地将指针设置为nil),你将不得不对特定类型做一个类型断言(*exec.Error)然后检查它是否为零,否则使用反射包.

使用反射来检查nil的示例:

func IsNil(i interface{}) bool {
    // Check if it is an actual nil-value
    if i == nil {
        return true
    }

    v := reflect.ValueOf(i)
    switch v.Kind() {
        // Only the following kinds can be nil
        case reflect.Chan, reflect.Func, reflect.Interface, reflect.Map, reflect.Ptr, reflect.Slice:
        return v.IsNil()        
    }

    return false
}
Run Code Online (Sandbox Code Playgroud)

工作示例:http://play.golang.org/p/rpG1PVTwwM

您可以在此处找到有关它的讨论:https://groups.google.com/forum/#!topic/golang -nuts/QzVDKv7p0Vs