个人主页:个人主页
系列专栏:Golang基础
Go(又称Golang)是由Google开发的开源编程语言。它结合了静态类型的安全性和动态语言的灵活性,拥有高效的并发编程能力和简洁的语法。Go被设计用于构建可扩展、高性能的软件系统,具有优秀的内存管理和快速的编译速度,适用于Web开发、系统编程和云计算等领域。
Go语言里面定义变量有多种方式。
使用 var 关键字是Go最基本的定义变量方式,与C语言不同的是Go把变量类型放在变量名后面:
//定义一个名称为“name”,类型为“type”的变量
var name type
//定义三个类型都是“type”的变量
var name1, name2, name3 type
定义变量并初始化值:
//初始化“name”的变量为“v”值,类型是“type”
var name type = v
/*
定义三个类型都是“type”的变量,并且分别初始化为相应的值
name1为v1,name2为v2,name3为v3
*/
var name1, name2, name3 type= v1, v2, v3
是不是觉得上面这样的定义有点繁琐?
有一种写法可以让它变得简单一点。
可以直接忽略类型声明,那么上面的代码变成这样了:
//Go会根据其相应值的类型来初始化它们
var name1, name2, name3 = v1, v2, v3
觉得上面的还是有些繁琐,继续简化:
//编译器会根据初始化的值自动推导出相应的类型
name1, name2, name3 := v1, v2, v3
现在是不是看上去非常简洁了?
:=
这个符号直接取代了 var 和 type ,这种形式叫做简短声明。
不过它有一个限制,那就是它只能用在函数内部,在函数外部使用则会无法编译通过,所以一般用 var 方式来定义全局变量。
_
(下划线)是个特殊的变量名,任何赋予它的值都会被丢弃。
在这个例子中,将值 35 赋予 b ,并同时丢弃 34 :
_, b := 34, 35
Go对于已声明但未使用的变量会在编译阶段报错,比如下面的代码就会产生一个错误:声明了 i 但未使用。
package main
func main() {
var i int
}
所谓常量,也就是在程序编译阶段就确定下来的值,而程序在运行时无法改变该值。
在Go程序中,常量可定义为数值、布尔值或字符串等类型。
它的语法如下:
const Name = v
//如果需要,也可以明确指定常量的类型
const Pi float32 = 3.1415926
下面是一些常量声明的例子:
const Pi = 3.1415926
const i = 10000
const MaxThread = 10
const prefix = "astaxie_"
Go 常量和一般程序语言不同的是,可以指定相当多的小数位数(例如200位),若指定给 float32 自动缩短为 32bit ,指定给 float64 自动缩短为 64bit ,详情参考 Go编程语言规范 。
在Go中,布尔值的类型为 bool ,值是 true 或 false ,默认为 false 。
var flag bool // 全局变量声明
var flag1, flag2 = true, false // 忽略类型的声明
func test() {
var flag3 bool // 一般声明
flag4 := false // 简短声明
flag4 = true // 赋值操作
}
整数类型有无符号和有符号两种。
Go同时支持 int 和 uint ,这两种类型的长度相同,但具体长度取决于不同编译器的实现。
Go里面也有直接定义好位数的类型: rune , int8 , int16 , int32 , int64 和 byte , uint8 , uint16 , uint32 , uint64 。
其中 rune 是 int32 的别称, byte 是 uint8 的别称。
需要注意的一点是,这些类型的变量之间不允许互相赋值或操作,不然会在编译时引起编译器报错。
如下的代码会产生错误:invalid operation: a + b (mismatched types int8 and int32)
var a int8
var b int32
c := a + b
另外,尽管 int 的长度是 32bit , 但 int 与 int32 并不可以互用。
浮点数的类型有 float32 和 float64 两种(没有 float 类型),默认是 float64 。
Go还支持复数。它的默认类型是 complex128 (64位实数+64位虚数)。
如果需要小一些的,也有complex64 (32位实数+32位虚数)。复数的形式为 RE + IMi ,其中 RE 是实数部分, IM 是虚数部分,
而最后的 i 是虚数单位。下面是一个使用复数的例子:
var c complex64 = 5+5i
//output: (5+5i)
fmt.Printf("Value is: %v", c)
Go中的字符串都是采用 UTF-8 字符集编码。字符串是用一对双引号 " "
或反引号 ``
括起来定义,
它的类型是 string 。
var s1 string // 声明变量为字符串的一般方法
var s2 string = "" // 声明了一个字符串变量,初始化为空字符串
func test() {
no, yes, or := "no", "yes", "or" // 简短声明,同时声明多个变量
s := "Konichiwa" // 同上
s = "Bonjour" // 常规赋值
}
在Go中字符串是不可变的,例如下面的代码编译时会报错:cannot assign to s[0]
var s string = "hello"
s[0] = 'c'
但如果真的想要修改怎么办呢?下面的代码可以实现:
s := "hello"
c := []byte(s) // 将字符串 s 转换为 []byte 类型
c[0] = 'c'
s2 := string(c) // 再转换回 string 类型
fmt.Printf("%s\n", s2)
Go中可以使用 + 操作符来连接两个字符串:
s := "hello,"
m := " world"
a := s + m
fmt.Printf("%s\n", a)
修改字符串也可写为:
s := "hello"
s = "c" + s[1:] // 字符串虽不能更改,但可进行切片操作
fmt.Printf("%s\n", s)
如果要声明一个多行的字符串怎么办?可以通过``
来声明:
m := `hello
world`
``
括起的字符串为 Raw 字符串,即字符串在代码中的形式就是打印时的形式,它没有字符转义,换行也将原样输出。例如本例中会输出:
hello
world
Go内置有一个 error 类型,专门用来处理错误信息,Go的 package 里面还专门有一个包 errors 来处理错误:
err := errors.New("error")
if err != nil {
fmt.Print(err)
}
在Go语言中,同时声明多个常量、变量,或者导入多个包时,可采用分组的方式进行声明。
例如下面的代码:
import "fmt"
import "os"
const I = 100
const Pi = 3.1415
const Prefix = "Go_"
var i int
var pi float32
var prefix string
可以分组写成如下形式:
import(
"fmt"
"os"
)
const(
I = 100
Pi = 3.1415
Prefix = "Go_"
)
var(
i int
pi float32
prefix string
)
Go里面有一个关键字 iota ,这个关键字用来声明 enum 的时候采用,它默认开始值是 0,const 中每增加一行加 1。
package main
import "fmt"
const (
x = iota // x == 0
y = iota // y == 1
z = iota // z == 2
w // 常量声明省略值时,默认和之前一个值的字面相同。这里隐式地说w = iota,因此w== 3。其实上面y和z可同样不用"= iota"
)
const v = iota // 每遇到一个const关键字,iota就会重置,此时v == 0
const h, i, j = iota, iota, iota //h=0,i=0,j=0 iota在同一行值相同
const (
a = iota //a=0
b = "B"
c = iota //c=2
d, e, f = iota, iota, iota //d=3,e=3,f=3
g = iota //g = 4
)
func main() {
fmt.Println(a, b, c, d, e, f, g, h, i, j, x, y, z, w, v)
}
除非被显式设置为其它值或 iota ,每个 const 分组的第一个常量被默认设置为它的0值,第二及后续的
常量被默认设置为它前面那个常量的值,如果前面那个常量的值是 iota ,则它也被设置为 iota 。
array 就是数组,它的定义方式如下:
var arr [n]type
在 [n]type 中, n 表示数组的长度, type 表示存储元素的类型。
对数组的操作和其它语言类似,都是通过 [ ] 来进行读取或赋值:
var arr [10]int // 声明了一个int类型的数组
arr[0] = 42 // 数组下标是从0开始的
arr[1] = 13 // 赋值操作
fmt.Printf("The first element is %d\n", arr[0]) // 获取数据,返回42
fmt.Printf("The last element is %d\n", arr[9]) //返回未赋值的最后一个元素,默认返回0
由于长度也是数组类型的一部分,因此 [3]int 与 [4]int 是不同的类型,数组也就不能改变长度。
数组之间的赋值是值的赋值,即当把一个数组作为参数传入函数的时候,传入的其实是该数组的副本,而不是它的指针。如果要使用指针,那么就需要用到后面介绍的 slice 类型了。
数组可以使用另一种 :=
来声明:
a := [3]int{1, 2, 3} // 声明了一个长度为3的int数组
b := [10]int{1, 2, 3} // 声明了一个长度为10的int数组,其中前三个元素初始化为1、2、3,其它默认为0
c := [...]int{4, 5, 6} // 可以省略长度而采用...的方式,Go会自动根据元素个数来计算长度
Go支持嵌套数组,即多维数组。比如下面的代码就声明了一个二维数组:
// 声明了一个二维数组,该数组以两个数组作为元素,其中每个数组中又有4个int类型的元素
doubleArray := [2][4]int{[4]int{1, 2, 3, 4}, [4]int{5, 6, 7, 8}}
// 上面的声明可以简化,直接忽略内部的类型
easyArray := [2][4]int{{1, 2, 3, 4}, {5, 6, 7, 8}}
在很多应用场景中,数组并不能满足需求。
在初始定义数组时,并不知道需要多大的数组,因此就需要“动态数组”,在Go里面这种数据结构叫 slice
slice 并不是真正意义上的动态数组,而是一个引用类型。
slice 总是指向一个底层 array , slice的声明也可以像 array 一样,只是不需要长度。
// 和声明array一样,只是少了长度
var fslice []int
接下来可以声明一个 slice ,并初始化数据,如下所示:
slice := []byte {'a', 'b', 'c', 'd'}
slice 可以从一个数组或一个已经存在的 slice 中再次声明。
slice 通过 array[i:j] 来获取,其中 i 是数组的开始位置, j 是结束位置,但不包含 array[j] ,它的长度是 j-i 。
// 声明一个含有10个元素元素类型为byte的数组
var ar = [10]byte {'a', 'b', 'c', 'd', 'e', 'f', 'g', 'h', 'i', 'j'}
// 声明两个含有byte的slice
var a, b []byte
// a指向数组的第3个元素开始,并到第五个元素结束,
a = ar[2:5]
//现在a含有的元素: ar[2]、ar[3]和ar[4]
// b是数组ar的另一个slice
b = ar[3:5]
// b的元素是:ar[3]和ar[4]
注意 slice 和 数组 在声明时的区别:声明数组时,方括号内写明了数组的长度或使用 … 自动计算长度,而声明 slice 时,方括号内没有任何字符。
slice 有一些简便的操作:
下面这个例子展示了更多关于 slice 的操作:
// 声明一个数组
var array = [10]byte{'a', 'b', 'c', 'd', 'e', 'f', 'g', 'h', 'i', 'j'}
// 声明两个slice
var aSlice, bSlice []byte
// 演示一些简便操作
aSlice = array[:3] // 等价于aSlice = array[0:3] aSlice包含元素: a,b,c
aSlice = array[5:] // 等价于aSlice = array[5:10] aSlice包含元素: f,g,h,i,j
aSlice = array[:] // 等价于aSlice = array[0:10] 这样aSlice包含了全部的元素
// 从slice中获取slice
aSlice = array[3:7] // aSlice包含元素: d,e,f,g,len=4,cap=7
bSlice = aSlice[1:3] // bSlice 包含aSlice[1], aSlice[2] 也就是含有: e,f
bSlice = aSlice[:3] // bSlice 包含 aSlice[0], aSlice[1], aSlice[2] 也就是含有:
d,e,f
bSlice = aSlice[0:5] // 对slice的slice可以在cap范围内扩展,此时bSlice包含:d,e,f,g,h
bSlice = aSlice[:] // bSlice包含所有aSlice的元素: d,e,f,g
slice 是引用类型,所以当引用改变其中元素的值时,其它的所有引用都会改变该值,例如上面的 aSlice 和 bSlice ,如果修改了 aSlice 中元素的值,那么 bSlice 相对应的值也会改变。
从概念上面来说 slice 像一个结构体,这个结构体包含了三个元素:一个指针,指向数组中 slice 指定的开始位置,长度,即 slice 的长度最大长度,也就是 slice 开始位置到数组的最后位置的长度
Array_a := [10]byte{'a', 'b', 'c', 'd', 'e', 'f', 'g', 'h', 'i', 'j'}
Slice_a := Array_a[2:5]
slice 有几个有用的内置函数:
注: append 函数会改变 slice 所引用的数组的内容,从而影响到引用同一数组的其它 slice 。
但当 slice 中没有剩余空间(即 (cap-len) == 0 )时,此时将动态分配新的数组空间。返回的 slice 数组指针将指向这个空间,而原数组的内容将保持不变;其它引用此数组的 slice 则不受影响。
从Go1.2开始 slice 支持了三个参数的 slice ,之前一直采用这种方式在 slice 或者 array 基础上来获取一个 slice
var array [10]int
slice := array[2:4]
这个例子里面slice的容量是 8,新版本里面可以指定这个容量:
slice = array[2:4:7]
上面这个的容量就是 7-2 ,即5。这样这个产生的新的 slice 就没办法访问最后的三个元素。
如果 slice 是这样的形式 array[:i:j] ,即第一个参数为空,默认值就是0 。
map 也就是Python中字典的概念,它的格式为:
map[keyType]valueType
看下面的代码, map 的读取和设置也类似 slice 一样,通过 key 来操作,只是 slice 的 index 只能是int 类型,而 map 多了很多类型,可以是 int ,可以是 string 及所有完全定义了 == 与 != 操作的类型。
// 声明一个key是字符串,值为int的字典,这种方式的声明需要在使用之前使用make初始化
var numbers map[string]int
// 另一种map的声明方式
numbers = make(map[string]int)
numbers["one"] = 1 //赋值
numbers["ten"] = 10 //赋值
numbers["three"] = 3
fmt.Println("第三个数字是: ", numbers["three"]) // 读取数据
// 打印出来如:第三个数字是: 3
这个 map 就像平常看到的表格一样,左边列是 key ,右边列是值
使用 map 过程中需要注意的几点:
map 的初始化可以通过 key:val 的方式初始化值,同时 map 内置有判断是否存在 key 的方式。
通过 delete 删除 map 的元素:
// 初始化一个字典
rating := map[string]float32{"C":5, "Go":4.5, "Python":4.5, "C++":2 }
// map有两个返回值,第二个返回值,如果不存在key,那么ok为false,如果存在ok为true
csharpRating, ok := rating["C#"]
if ok {
fmt.Println("C# is in the map and its rating is ", csharpRating)
} else {
fmt.Println("We have no rating associated with C# in the map")
}
delete(rating, "C") // 删除key为C的元素
上面说过了, map 也是一种引用类型,如果两个 map 同时指向一个底层,那么一个改变,另一个也相应的改变:
m := make(map[string]string)
m["Hello"] = "Bonjour"
m1 := m
m1["Hello"] = "Salut" // 现在m["hello"]的值已经是Salut了
make 用于内建类型( map 、 slice 和 channel )的内存分配。
new 用于各种类型的内存分配。
内建函数 new 本质上说跟其它语言中的同名函数功能一样: new(T) 分配了零值填充的 T 类型的内存空间,并且返回其地址,即一个 *T 类型的值。用Go的术语说,它返回了一个指针,指向新分配的类型 T 的零值。有一点非常重要:
new 返回指针
内建函数 make(T, args) 与 new(T) 有着不同的功能,make 只能创建 slice 、 map 和 channel ,并且返回一个有初始值(非零)的 T 类型,而不是 *T 。
本质来讲,导致这三个类型有所不同的原因是指向数据结构的引用在使用前必须被初始化。
例如,一个 slice ,是一个包含指向数据(内部 array )的指针、长度和容量的三项描述符;在这些项目被初始化之前, slice 为 nil 。对于 slice 、 map 和 channel 来说, make 初始化了内部的数据结构,填充适当的值。
make 返回初始化后的(非零)值
关于“零值”,所指并非是空值,而是一种“变量未填充前”的默认值,通常为 0。
此处罗列部分类型的“零值”:
int 0
int8 0
int32 0
int64 0
uint 0x0
rune 0 //rune的实际类型是 int32
byte 0x0 // byte的实际类型是 uint8
float32 0 //长度为 4 byte
float64 0 //长度为 8 byte
bool false
string ""
Go之所以会那么简洁,是因为它有一些默认的行为: