引言
在Go语言中,变量的初始化是编写程序时经常遇到的重要操作之一。
通过正确地初始化变量,我们可以确保程序的正确性和可读性,并且能够避免一些常见的错误。
Go语言提供了多种初始化变量的方法,包括使用var关键字、短变量声明以及在声明时提供初始值等方式。
本文将介绍如何在Go语言中进行变量的初始化,并提供一些示例以更好地理解这些概念。
举例
在Go语言中,变量的初始化可以使用var关键字或短变量声明,并且可以同时声明和初始化变量。下面是一些示例:
1.使用var关键字声明和初始化变量:
var x int = 10 var str string = "Hello, world!"
2.使用短变量声明方式:
x := 10 str := "Hello, world!"
3.多个变量同时声明和初始化:
var a, b, c int = 1, 2, 3 var str1, str2 string = "foo", "bar"
4.如果声明变量时未提供初始值,则变量会被初始化为其类型的零值:
var num int // num被初始化为0 var str string // str被初始化为空字符串""
5.在函数内部,可以使用短变量声明的方式初始化变量:
func main() { x := 10 str := "Hello, world!" fmt.Println(x, str) }
在Go中,变量的初始化是一种常见且重要的操作,它使得代码更加清晰易懂,并且有助于避免潜在的错误。
多个变量同时赋值
在Go语言中,可以使用多个变量同时进行赋值。这可以通过使用简短变量声明(short variable declaration)或使用赋值语句来实现。
1.使用简短变量声明方式:
x, y := 10, 20 str1, str2 := "Hello", "World"
2.使用赋值语句同时给多个变量赋值:
var a, b int a, b = 10, 20
3.交换两个变量的值:
x, y := 10, 20 x, y = y, x // 交换x和y的值
4.同时声明和赋值多个变量:
var ( name = "John" age = 30 email = "john@example.com" )
Go 语言还提供了两个专门用于数据初始化的内建函数 new 和 make。
1. 内建函数 new
new 函数用于为值分配内存。与其他编程语言(如 Java)中的 new 不同的是,它并不会去初始化分配到的内存,而只会清零它。因此调用表达式 new(T) 被求值时,所做的是为 T 类型的新值分配并清零一块内存空间,然后将这块内存空间的地址作为结果返回。而这个结果就是指向这个新的 T 类型值的指针值。它的类型为 *T。实际上这个 new 函数返回的 *T 类型值总会指向一个 T 类型的零值。
例如,调用表达式 new(string) 的求值结果指向的是一个 string 类型的零值 "",而调用表达式 new([3]int) 的求值结果指向的则是一个 [3]int 类型的零值 [3]int{0, 0, 0}。
正因为有这种干净的内存分配策略,使得我们可以在用内建函数 new 创建某个数据类型的新值之后立刻就可以拿来使用,而不用担心在这个值中会遗留某些初始化的痕迹。
下面我们以标准库代码包 bytes 中的结构体类型 Buffer 为例介绍以下:
bytes.Buffer 是一个尺寸可变的字节缓冲区。它的零值就是一个立即可用的空缓冲区。因此,调用表达式 new(bytes.Buffer) 的求值结果就是一个指向一个空缓冲区的指针值。然后,我们就可以立即在这个缓冲区上进行读写操作了。
标准库代码包 sync 中的结构体类型 Mutex 也是一个可以 new 后即用的数据类型。它的零值就是一个处于未锁定状态的互斥量。
内建函数 new 的这种特性为我们提供了一个关于自定义数据类型的可参考的设计规则:
例如,在我们自定义一个结构体类型的时候就要考虑到,在其中的每个字段的值都分别为对应类型的零值的时候,这个结构体值就应该已经处于可用的状态。这样,我们在 new 它的时候就能够得到一个立即可用的值的指针值,而不需要再做额外的初始化。
当然,在感觉一个类型的零值还无法让它变得可用的时候,我们可以使用相应的字面量来达到分配内存空间并初始化值的目的。
前面我们已经讲过,我们可以在字面量中灵活的指定新值中的每一个元素的值。但是要注意,字面量所代表的是该类型的值,而不是指向该类型值的指针值。 因此,我们在将它们与调用 new 函数的调用表达式做等价替换的时候,还需要在字面量的前面加入取址操作符 & 以表示指向该类型值的指针值。
2. 内建函数 make
与 new 不同的是,make 函数只能被用于创建 切片类型、字典类型 和 通道类型 的值,并返回一个已被初始化的(即非零值的)的对应类型的值。
那 make 函数为啥要这样做呢?
在之前的博文中我们介绍过,这 3 个复合类型都是引用类型。在它们的每一个值的内部都会保持着一个对某个底层数据结构值的引用。如果不对它们的值进行初始化,那么其中的这种引用关系是不会被建立起来的,同时相关的内部值也会不正确。在这种情况下,该类型的值也就不能够被使用,因为它们是不完整的,还处于未就绪的状态。这就意味着,在创建这 3 个引用类型的值的时候,必须将内存空间分配和数据初始化这两个步骤绑定在一起。也正是为了保证这些值的可用性,切片类型、字典类型 和 通道类型的零值都是 nil,不是那个未被初始化的值。因此,当我们 new 这 3 个引用类型并想创建它们的值的时候,得到的却是一个指向空值 nil 的指针值。
除此之外,内建函数 make 所接受的参数也与 new 函数有所不同。make 函数除了会接受一个表示目标类型的类型字面量之外,还会接受一个或两个额外的参数。
对于 切片类型 来说,我们可以在把新值的长度和容量也传递给 make 函数。例如:
// 调用表达式创建了一个新的 []int 类型的值,这个值的长度为10、容量为100 make([]int, 10, 100)
当然,我们也可以省略掉最后一个参数,即不指定新值的容量。这种情况下,该值的容量会与其长度一致。示例如下:
// 变量s的类型是[]int的,而长度和容量都是10。 s := make([]int,10)
在使用 make 函数初始化一个切片值的过程中,该值会引用一个长度与其容量相同且元素类型与其元素类型一致的数组值。这个数组值就是该切片值的底层数组。该数组值中的每个元素都是当前元素类型的零值。但是,切片值只会展现出数量与其长度相同的元素。因此,调用表达式 make([]int,10,100) 所创建并初始化的值就是 []int{0 0 0 0 0 0 0 0 0 0}。
我们在使用 make 函数创建字典类型的值的时候,也可以指定其底层数据结构的长度。但是,该字典值只会展示出我们明确放入的键值对。例如:
// 调用如下表达式,所创建和初始化的值会是 map[string]int make(map[string]int,100) // 也可以忽略掉那个用于表示底层数据结构长度的参数 make(map[ string]int)
虽然我们也可以忽略掉那个用于表示底层数据结构长度的参数 ,但是还是建议:
应该在性能敏感的应用场景下,根据这个字典值可能包含的键值对的数量以及放入它们的时间,仔细地设置该长度参数。
最后我们简单介绍以下对于 通道类型(Channel) 的值的数据初始化。到此目前为止,我们还没有对通道类型进行过任何说明。不过,在后续的博文中,我们会详细讲解这个特殊的引用类型。
// 使用make函数创建一个通道类型的值 make(chan int, 10)
其中的第一个参数表示的是 通道的类型,而第二个参数则表示该 通道的长度。与字典类型相同,第二个参数也可以被忽略掉。不过忽略它的含义却与针对字典类型的情况有着很大的不同,这些我们会在后面的博文中专门讲解。
注意: make 函数只能被应用在引用类型的值的创建上。并且,它的结果是第一个参数所代表的类型的值,而不是指向这个值的指针值。
如果我们想要获得该指针值的话,只能在调用 make 函数的表达式的求值结果之上应用取址操作符 &,像这样:
m := make(map[string]int, 100) mp := &m
到目前为止,我们已经介绍了 3 种创建值的的方法:
- 使用 字面量
- 调用内建函数 new
- 调用内建函数 make
它们适用于不同的应用场景。当然,在某些应用场景中、我们可以有多种选择:
- 在创建一个切片类型的值的时候,我们既可以使用字面量也可以使用 make 函数。这种选择的结果往往取决于我们是否需要定制切片值中的某个或某些元素值。
- 如果我们能够保证一个结构体类型的值在其中字段的值均为零值的情况下就能够处于可用状态的话,那么仅使用 new 函数来初始化它与使用字面量进行初始化是基本等价的。不过要注意,这两种方法产生的结果的类型是不同的。
总结
通过本文的介绍,相信已经了解了在Go语言中进行变量初始化的基本方法和注意事项。
无论是使用var关键字声明和初始化变量,还是通过短变量声明方式,都可以轻松地初始化变量并开始编写Go程序。
在编写代码时,始终确保正确地初始化变量是非常重要的,这有助于代码的清晰度和可维护性。