一、值、指针类型及引用类型

在Go中,一个变量持有的内容无非三种:值类型、指针类型及引用类型。值类型和指针类型可以通过取值和取址操作互为转换,而指针类型和引用类型常常被人混为一谈,下面分别说明这个三种类型概念。

1.值类型

Go语言的值类型有以下几种:

  • 基本数据类型(数值、布尔、字符、字节)
  • 数组array
  • 结构体struct(结构体会在面向“对象”专题展开)

一个存储值类型的变量,它在函数或方法的参数传递中是属于拷贝传递的,即:传递的值类型参数在函数或方法内部修改不会影响外部的变量数据,而只会影响其拷贝的数据。

2.指针类型和指针地址

上面我们提到值传递,在传递基本数据类型时,这种类型的变量只占1~8字节,可以说非常廉价,对性能影响可以忽略不计。但在传递数组和结构体时,如果数据体量较大,传递一个数组或结构体代价是比较大的。
在开发中会遇到这种需求:一个变量传递给一个函数或方法,希望函数内部对变量的修改可以影响外部,这时我们就需要把变量的内存地址作为参数传给函数或方法,这种传地址的方式就是传递指针类型或引用类型(下面提到),可见指针的本质就是内存地址,一个指针类型的变量存放的就是指向源数据的内存地址。

玩过C语言的都知道,指针是非常重要的概念,由于C语言中允许对指针进行运算,所以C中的指针操作有一定复杂度,对指针的管理特耗程序员心智,稍有不慎就会出错。Go对指针进行精简设计,只允许对指针进行取值和取地址操作,而且其最多支持二级指针,这大大简化了程序员的使用难度,且不像C那样容易出错。只要熟悉指针的本质,一样可以玩的溜!

一个值类型的变量都可以通过指针操作符获取数据的内存地址,而存放内存地址的变量的类型就是指针类型。

  1. var i = 1
  2. iPtr := &i
  3. ii := *iPtr
  4. fmt.Printf("i的类型为%T,值为%v\n", i, i)
  5. fmt.Printf("iPtr的类型为%T,值为%v\n", iPtr, iPtr)
  6. fmt.Printf("ii的类型为%T,值为%v\n", ii, ii)
  7. //i的类型为int,值为1
  8. //iPtr的类型为*int,值为0xc000116048
  9. //ii的类型为int,值为1

如上所见:i为存储int数据的值类型变量,对i取地址后存到变量iPtr,iPtr为存储指向int类型数据的指针,对iPtr指针类型取值后存到变量ii,ii为存储int数据的值类型变量

在对普通变量使用&操作符取地址获得这个变量的指针后,可以对指针使用*操作,也就是指针取值

  • &取址符可对任何变量使用,获取变量地址
  • *指针取值符只可对指针变量使用

3.引用类型

除值类型和指针类型,Go还内置了几种引用类型,所谓引用类型,是指当变量接收该类型的数据时,存储的是其内存head地址而非其数据本身。Go内置的引用类型为以下几种:

  • 切片slice
  • 映射map
  • 函数func
  • 接口interface
  • nil

由于存储引用类型的变量是地址数据,所以其在传递过程性能较高,需要注意的是,传递引用类型时,函数或方法内部对参数的修改会影响外部。

二、nil及零值

与其他语言一样,Go语言也有指代空值的标识符:nil。但其又不是简单含义的空值。Go官方说明了,nil是预定义的标识符,代表指针、通道、函数、接口、映射或切片的零值。简单来说,nil实际上并不是指针,除基本值类型外,任何未分配内存空间的变量声明都指向nil,即零号内存地址(0x0)。
例如:

  1. var a []int = nil
  2. fmt.Printf("a的类型为%T,地址为%p\n", a, a)
  3. //a的类型为[]int,地址为0x0

说到零值,也顺带提一下go中内置值类型的零值,基本值类型不能等于nil。

  1. //不能通过编译
  2. //var a int = nil
  3. //打印基本值类型的零值
  4. var aa int
  5. var bb bool
  6. var cc float64
  7. var dd [3]int
  8. type St struct {
  9. a string
  10. b bool
  11. c int64
  12. }
  13. var st St
  14. fmt.Printf("aa的类型为%T,值为:%v,地址为%p\n", aa, aa, &aa)
  15. fmt.Printf("bb的类型为%T,值为:%v,地址为%p\n", bb, bb, &bb)
  16. fmt.Printf("cc的类型为%T,值为:%v,地址为%p\n", cc, cc, &cc)
  17. fmt.Printf("dd的类型为%T,值为:%v,地址为%p\n", dd, dd, &dd)
  18. fmt.Printf("st的类型为%T,值为:%v,地址为%p\n", st, st, &st)
  19. //aa的类型为int,值为:0,地址为0xc00008e018
  20. //bb的类型为bool,值为:false,地址为0xc00008e020
  21. //cc的类型为float64,值为:0,地址为0xc00008e028
  22. //dd的类型为[3]int,值为:[0 0 0],地址为0xc000096020
  23. //st的类型为base.St,值为:{ false 0},地址为0xc000088020

可见基本值类型声明后就已经分配内存地址,并赋予零值。