1 概述

Go语言作为类C语言,支持常规的基础数据类型的的同时,支持常用的高级数据类型。他们是:

  • 整数,int,uint,int8,uint8,int16,uint16,int32,uint32,int64,uint64
  • 浮点(小数),float32,float64
  • 复数,complex
  • 字符,byte,rune
  • 字符串,string
  • 布尔,bool
  • 指针,pointor
  • 数组,array
  • nil
  • 切片,slice。(引用类型)
  • 映射,map
  • 结构体,struct
  • 接口,interface
  • 函数,func()

本篇相当于一个类型索引。包含了Go语言中的全部类型。具体每种类型的操作,请参见具体章节!

2 整数,int

支持的整型分类很细,主要差异是占用的存储空间不同。基于有无符号列出为:
有符号的:int,int8,int16,int32,int64
无符号的:uint,uint8,uint16,uint32,uint64

其中int, unit是基于处理器架构的。意味着会根据处理器来确定是32bit还是64bit。
使用时,常常使用int。或者使用int32保证兼容。
在类型推导时,推导的整型为int,其他长度整型需要在定义变量时强制声明。
示例:

42
1024
-36

整型的默认值为0。

3 浮点数,float

支持IEEE-754标准浮点数。支持32bit和64bit的浮点数float32和float64。
在类型推导时,推导的浮点型为float64。float32需要在定义变量时强制声明。
示例:

1.0
3.14
0.618
4.2E10 // 4.2*10^10

浮点数的默认值为0。

4 复数,complex32,complex64

原生支持复数。
支持32bit+32bit复数complex64,和64bit+64bit的复数complex128。

5 字符,byte,rune

使用单引号定义字符型数据,字符指的是单个字符。有两种byte和rune:
byte:单字节字符,是uint8的别名。用于存储ASCII字符集字符
rune:多字节字符,是int32的别名。用于存储unicode字符集字符。
在类型推导时,推导的字符型为rune。byte需要在定义变量时强制声明。
示例:

'c'
'康'

字符的默认值是0。字符的本质就是整数型,根据字符集得到对应的字符。

6 字符串,string

原生支持字符串。使用双引号("")或反引号(``)定义,内部是utf-8编码。
双引号:"", 用于单行字符串。
反引号:``,用于定义多行字符串,内部会原样解析。
示例:

// 单行
"心有猛虎,细嗅蔷薇"
// 多行
`
大风歌
大风起兮云飞扬。
威加海内兮归故乡。
安得猛士兮守四方!
`

字符串支持转义字符,列表如下:

  • \r 回车符(返回行首)
  • \n 换行符(直接跳到下一行的同列位置)
  • \t 制表符
  • \' 单引号
  • \" 双引号
  • \\ 反斜杠
  • \uXXXX Unicode字符码值转义

7 布尔型,bool

布尔型的值只可以是常量 true 或者 false。
示例:

var (
  isFinished = false
  )

8 指针,pointer

指针类型用于存储变量地址。使用运算符 & , *完成操作。
使用运算符 p:=&v 可以获取v变量的地址。p中的数据格式类似0xc0000b9528,是一个内存地址的16进制表示。
使用运算符 *p 可以获取指针p指向变量v的值。
如图所示:

在Go语言中,指针主要用于:

  • 类型指针,对指针指向的数据进行修改。函数调用时传递参数使用指针,避免值传递拷贝额外数据。注意类型指针不能进行偏移和运算。
  • 切片,由指向起始元素的原始指针、元素数量和容量组成。
    示例:
    p := 42
    pv := &p
    *pv ++
    // p == 43

9 数组,array

数组是一段固定长度的连续内存区域。是具有相同类型数据元素序列。元素类型支持任意内置类型。
数组从声明时就确定长度,可以修改数组元素,但是数组不可修改长度。
使用 [长度]类型 进行数组的声明。
示例:

// 默认值初始化
var nums [4]int // [0 0 0 0]
// 指定初始值
var position = [4]string{"east", "south", "west", "north"}
// 自动确定长度
var position = [...]string{"east", "south", "west", "north"}

会使用类型默认值初始化元素。
数组不是引用类型,变量间为值传递。

可以使用range配合循环结构完成遍历,示例如下:

for k, v := range position {
    fmt.Println(k, v)
}
// 结果
0 east
1 south
2 west
3 north

10 nil

nil,可以理解为未初始化引用。是以下类型的默认初始值:

  • pointers -> nil
  • slices -> nil
  • maps -> nil
  • channels -> nil
  • functions -> nil
  • interfaces -> nil

11 切片,slice

切片是一个拥有相同类型元素的可变长度的序列。与数组的固定长度不同,切片也被称之为动态数组。
Go提供了4中方案定义切片:

  make ( []Type ,length, capacity )
  make ( []Type, length)
  []Type{}
  []Type{value1 , value2 , ... , valueN }

或者从数组或切片生成新切片:
slice [开始索引:结束索引]

  • slice 表示目标切片对象。
  • 开始索引和结束索引对应目标切片的索引。
  • 不包含结束索引对应的元素
  • 缺省开始索引,表示从头到结束索引。
  • 缺省结束索引,表示从开始索引到末尾。
  • 两者同时缺省时,全部切取,与切片本身等效。
  • 两者同时为0时,等效于空切片,一般用于切片复位。
    var arr = [4]string{"a", "b", "c", "d"}
    var sli = arr[1:3] // ["b", "c"]

    以从数组创建切片为例,理解切片,定义语法如上所示,下图为slice:

切片的实现是由一个底层数组以及其上面的动态位置,尺寸来实现。由内部由指向起始元素的指针、元素数量length和容量capacity组成。其中:

  • 指针ptr,用于指向切片在底层数组的起始位置。
  • 尺寸len,用于记录切片内元素数量。
  • 容量cap,当前切片最大容量,也就是底层数组的容量。可以动态分配。

切片为引用类型。
切片的默认初始值为nil。
切片支持: len()尺寸, cap()容量, append()追加元素等操作。详见切片的使用。

12 映射,map

Go语言中的键值对(key->value)集合,称之为映射map。
创建语法:
var m = map[key_type]value_type{key1: value1, key2: value2}
var m = make(map[key_type]value_type)
示例,字符串型下标,字符串型值:

var m = make(map[string]string) //make()会分配内存空间,初始化。
m["east"] = "东"
m["west"] = "西"
fmt.Println(m["east"]) // 东
fmt.Println(m["west"]) // 西

// map演示
var m = map[string]string{"east": "东", "west": "西"}

支持遍历操作,使用range:

for k, v := range mapVar {
    fmt.Println(k, v)
}

映射是引用类型。
详见切片操作。

13 结构体,struct

Go语言使用结构体来描述现实业务逻辑中实体。是自定义类型。结构体是由一系列数据构成的数据集合,一系列数据类型可以不同。
定义结构体,使用struct关键字:
type 结构体名 struct {
成员1 类型
成员2 类型

}
示例:

// 定义Product结构
type Product struct {
  // 两个成员
  name  string
  price float64
}

// 构造函数
func newProduct(name string, price float64) *Product {
  return &Product{name, price}
}

// 成员方法,接收器方式
func (p *Product) getName() string {
  return "《" + p.name + "》"
}

// 方法2
func (p *Product) setPrice(price float64) *Product {
  p.price = price
  return p
}

func main() {
  // 构造Product型数据p1
  var p1 = newProduct("BlockChain", 42.5)
  // 访问成员
  fmt.Println(p1.name)
  // 通过接收器访问方法
  fmt.Println(p1.getName())
  var p2 = newProduct("GoLang", 30.5)
  p2.setPrice(44.5)
  fmt.Println(p2.price)
}

详见结构体操作。

14 接口,interface

接口是一种协议,用来规范方法的调用和定义的协议,目的是保证设计的一致性,便于模块化开发以及通讯。Go语言中,也视为一种类型。
定义语法:
type 接口名 interface {
方法1( 参数列表 ) 返回值类型列表
方法2( 参数列表 ) 返回值类型列表

}
可以理解成没有方法体的方法。
示例:

// 定义Product结构
type Product struct {
  // 两个成员
  name  string
  price float64
  // say   func()
}

// 构造函数
func newProduct(name string, price float64) *Product {
  return &Product{name, price}
}

// 成员方法,接收器方式
func (p *Product) getName() string {
  return "《" + p.name + "》"
}

// 方法2
func (p *Product) setPrice(price float64) *Product {
  p.price = price
  return p
}

// 定义接口
type ProductInterface interface {
  getName() string
  setPrice(price float64) *Product
}

func main() {
  // 构造Product型数据p1
  var p1 = newProduct("BlockChain", 42.5)
  // pi为接口,为其赋值p1,就意味着使用接口规范p1,若p1不满足接口定义则出错
  var pi ProductInterface = p1
  // 利用接口调用方法
  fmt.Println(pi.getName())
}

详见接口操作

15 函数,func()

Go语言中,函数可以作为数据存储变量中,此时变量为函数类型func()。可以通过该变量访问到这个函数。可以用在结构体成员定义上。
语法示例:

func sayHello(name string) {
  fmt.Println("Hello, ", name)
}

func main() {
  // 函数本身就是函数类型
  fmt.Printf("%T(%v)\n", sayHello, sayHello) // func(string)(0x48fe20)
  // 赋值给变量
  var f = sayHello
  fmt.Printf("%T(%v)\n", f, f) // func(string)(0x48fe20)
  // 匿名函数也是函数类型
  var af = func() {
  }
  fmt.Printf("%T(%v)\n", af, af) // func()(0x490080)
}

参见函数使用。

完!
原文出自:小韩说课
微信关注:小韩说课