Go语言nil:空值/零值
本文内容纲要:
-nil标识符是不能比较的
-nil不是关键字或保留字
-nil没有默认类型
-不同类型nil的指针是一样的
-不同类型的nil是不能比较的
-两个相同类型的nil值也可能无法比较
-不同类型的nil值占用的内存大小可能是不一样的
在Go语言中,布尔类型的零值(初始值)为false,数值类型的零值为0,字符串类型的零值为空字符串""
,而指针、切片、映射、通道、函数和接口的零值则是nil。
nil是Go语言中一个预定义好的标识符,有过其他编程语言开发经验的开发者也许会把nil看作其他语言中的null(NULL),其实这并不是完全正确的,因为Go语言中的nil和其他语言中的null有很多不同点。
下面通过几个方面来介绍一下Go语言中nil。
nil标识符是不能比较的
packagemain
import(
"fmt"
)
funcmain(){
fmt.Println(nil==nil)
}
运行结果如下所示:
PSD:\code>gorun.\main.go
#command-line-arguments
.\main.go:8:21:invalidoperation:nil==nil(operator==notdefinedonnil)
//这点和python等动态语言是不同的,在python中,两个None值永远相等。
>>>None==None
True
从上面的运行结果不难看出,==对于nil来说是一种未定义的操作。
nil不是关键字或保留字
nil并不是Go语言的关键字或者保留字,也就是说我们可以定义一个名称为nil的变量,比如下面这样:
varnil=errors.New("mygod")
虽然上面的声明语句可以通过编译,但是并不提倡这么做。
nil没有默认类型
packagemain
import(
"fmt"
)
funcmain(){
fmt.Printf("%T",nil)
print(nil)
}
运行结果如下所示:
PSD:\code>gorun.\main.go
#command-line-arguments
.\main.go:9:10:useofuntypednil
不同类型nil的指针是一样的
packagemain
import(
"fmt"
)
funcmain(){
vararr[]int
varnum*int
fmt.Printf("%p\n",arr)
fmt.Printf("%p",num)
}
运行结果如下所示:
PSD:\code>gorun.\main.go
0x0
0x0
通过运行结果可以看出arr和num的指针都是0x0。
不同类型的nil是不能比较的
packagemain
import(
"fmt"
)
funcmain(){
varmmap[int]string
varptr*int
fmt.Printf(m==ptr)
}
运行结果如下所示:
PSD:\code>gorun.\main.go
#command-line-arguments
.\main.go:10:20:invalidoperation:arr==ptr(mismatchedtypes[]intand*int)
两个相同类型的nil值也可能无法比较
在Go语言中map、slice和function类型的nil值不能比较,比较两个无法比较类型的值是非法的,下面的语句无法编译。
packagemain
import(
"fmt"
)
funcmain(){
vars1[]int
vars2[]int
fmt.Printf(s1==s2)
}
运行结果如下所示:
PSD:\code>gorun.\main.go
#command-line-arguments
.\main.go:10:19:invalidoperation:s1==s2(slicecanonlybecomparedtonil)
通过上面的错误提示可以看出,能够将上述不可比较类型的空值直接与nil标识符进行比较,如下所示:
packagemain
import(
"fmt"
)
funcmain(){
vars1[]int
fmt.Println(s1==nil)
}
nil是map、slice、pointer、channel、func、interface的零值
packagemain
import(
"fmt"
)
funcmain(){
varmmap[int]string
varptr*int
varcchanint
varsl[]int
varffunc()
variinterface{}
fmt.Printf("%#v\n",m)
fmt.Printf("%#v\n",ptr)
fmt.Printf("%#v\n",c)
fmt.Printf("%#v\n",sl)
fmt.Printf("%#v\n",f)
fmt.Printf("%#v\n",i)
}
运行结果如下所示:
PSD:\code>gorun.\main.go
map[int]string(nil)
(*int)(nil)
(chanint)(nil)
[]int(nil)
(func())(nil)
<nil>
零值是Go语言中变量在声明之后但是未初始化被赋予的该类型的一个默认值。
不同类型的nil值占用的内存大小可能是不一样的
一个类型的所有的值的内存布局都是一样的,nil也不例外,nil的大小与同类型中的非nil类型的大小是一样的。但是不同类型的nil值的大小可能不同。
packagemain
import(
"fmt"
"unsafe"
)
funcmain(){
varp*struct{}
fmt.Println(unsafe.Sizeof(p))//8
vars[]int
fmt.Println(unsafe.Sizeof(s))//24
varmmap[int]bool
fmt.Println(unsafe.Sizeof(m))//8
varcchanstring
fmt.Println(unsafe.Sizeof(c))//8
varffunc()
fmt.Println(unsafe.Sizeof(f))//8
variinterface{}
fmt.Println(unsafe.Sizeof(i))//16
}
运行结果如下所示:
PSD:\code>gorun.\main.go
8
24
8
8
8
16
具体的大小取决于编译器和架构,上面打印的结果是在64位架构和标准编译器下完成的,对应32位的架构的,打印的大小将减半。
本文内容总结:nil标识符是不能比较的,nil不是关键字或保留字,nil没有默认类型,不同类型nil的指针是一样的,不同类型的nil是不能比较的,两个相同类型的nil值也可能无法比较,不同类型的nil值占用的内存大小可能是不一样的,
原文链接:https://www.cnblogs.com/lurenq/p/12013168.html