GO在nil判断等方面容易混淆。
nil
GO的nil是带类型的,可以参考相关的文章。看下面的代码:
import (
"io"
"fmt"
)
type MyWriter struct {
}
func (w *MyWriter) Write(p []byte) (n int, err error) {
return
}
func main() {
var m *MyWriter
fmt.Println("ptr m==nil?", bool(m==nil))
var o io.Writer = m
fmt.Println("interface o==nil?", bool(o==nil))
if a,ok := o.(*MyWriter); true {
fmt.Println("dynamic covert ok?", ok, "and a==nil?", bool(a==nil))
}
o = func()io.Writer{
var m *MyWriter
return m
}()
fmt.Println("return o==nil?", bool(o==nil))
func(o io.Writer) {
fmt.Println("params o==nil?", bool(o==nil))
}(m)
os.Exit(0)
}
结果比较奇特:
ptr m==nil? true
interface o==nil? false
dynamic covert ok? true and a==nil? true
return o==nil? false
params o==nil? false
简单的规则就是一条,nil是有类型的:
var m *MyWriter = nil
// 实际上是带类型的(type, value):
var m *MyWriter = ((*MyWriter)nil, nil)
因此,在比较的时候,都是带类型比较:
var m *MyWriter = nil
if m == nil {
// 相当于:
if m == ((*MyWriter)nil, nil) {
// 这个地方当然是相等的,类型等,值也等。
但是换成其他的类型就不等了:
var m *MyWriter = nil
var o io.Writer = m
//这时候的o实际上是:((*MyWriter)nil,nil)
if o == ((io.Writer)nil, nil) {
这个在返回值和参数时就非常重要,特别是参数:
func(o io.Writer) {
fmt.Println("params o==nil?", bool(o==nil))
}(m)
大部分的时候,传递过来的肯定不是接口,而是某个指针,除非直接调用,否则都是非nil的,只有下面这种直接调用情况是nil:
func(o io.Writer) {
fmt.Println("params o==nil?", bool(o==nil))
}(nil)
这个确实比较容易混淆。