Go语言 数组、切片、map的区别

简介: Go语言 数组、切片、map的区别

本篇只进行比较!


类型比较


数组:值类型

切片:引用类型

map:引用类型


语法比较


数组的语法:var name [5]string []必须指定长度数字

var a  [10]int //声明数组

切片的语法:var name []string

var b []int    //声明切片

map的语法:var map变量名 map[key类型]vlaue类型 []中必须指定类型

var c map[string]int   //声明map类型


长度和容量规则


数组:固定长度,无容量。数组的长度在声明时就要给出

切片:动态长度,有容量,容量可以理解成可达到的最大长度

切片可以由数组构造来,不改变长度的情况下,共享内存,即切片的第一个元素发生变化,数组的第一个元素也将发生变化。

package main
import "fmt"
func main() {
  a := [10]int{1,2,3,4,5}
  b := a[:]  //声明切片b由数组a构造
  b[4] =6 //只修改了切片b的第四个元素,数组a也将发生改变
  fmt.Println(a)
  fmt.Println(b)
}

输出结果:

[1 2 3 4 6 0 0 0 0 0]
[1 2 3 4 6 0 0 0 0 0]

上面不发生变化的只发生再数组转换成切片的情况下,如果数组与数组之间的赋值,是不会共享内存的

package main
import "fmt"
func main() {
  a := [...]string{"USA", "China", "India", "Germany", "France"}
  b := a // a的一个副本被分配给b
  b[0] = "Singapore"
  fmt.Println("a is ", a)
  fmt.Println("b is ", b)
  fmt.Printf("%T%T",a,b)
}

输出结果:

a is  [USA China India Germany France]
b is  [Singapore China India Germany France]
[5]string[5]string //都是数组类型,同类型

切片的长度可使用append参数,改变后的切片超出容量将不共享内存

package main
import "fmt"
func main() {
  a := [...]int{1,2,3,4,5} //a数组
  b := []int{6,7,8,9,10}   //b切片
  c := a[:]               //声明切片b由数组a构造
  c = append(c,b...) //必须是切片类型才能追加,并第二个切片加“...”
  fmt.Println(a)
  fmt.Println(b)
  fmt.Println(c)
}

输出结果:

[1 2 3 4 5]
[6 7 8 9 10]
[1 2 3 4 5 6 7 8 9 10]

map:动态长度

map是无序的,我们无法决定它的返回顺序,这是因为 Map 是使用 hash 表来实现的。


声明比较


package main
import "fmt"
func main() {
  var a  [10]int //声明数组
  var b []int    //声明切片
  var c map[string]int   //声明map类型
  fmt.Println(a)
  fmt.Println(b == nil)
  fmt.Println(c == nil)
}

输出结果:

[0 0 0 0 0 0 0 0 0 0]
true
true

数组是不能被判断是否是nil,因为数组必须在声明时申请内存

切片和map可以被判断是否是nil


初始化比较


切片和map是可以被make初始化的,数组不行,数组必须在声明时初始化。

初始化之后才能有内存地址

package main
import "fmt"
func main() {
  a := make([]int,5,10)
  b := make(map[string]int,10)
  fmt.Printf("%T\n",a)
  fmt.Printf("%T",b)
}

输出结果:

[]int
map[string]int

另外,new和make都是用于初始化的,它们两个区别在于new只能初始化值类型(int、string等),make只能初始化引用类型(切片、map、channel)


相互转换


数组可以转换成切片

package main
import "fmt"
func main() {
  var a  [10]int
  var b  = a[:] //不加[:]代表转化成数组
  fmt.Println(a)
  fmt.Println(b)
}

输出结果:

[0 0 0 0 0 0 0 0 0 0]
[0 0 0 0 0 0 0 0 0 0]


修改值的区别

package main
import "fmt"
func main() {
  a := make([]int,5,10)
  b := make(map[string]int,10)
  c := [5]int{}
  c[1]=10 //把数组c第一个元素修改成10
  a[1] = 10 //把切片a第一个元素修改成10
  b["Go语言"] = 1 // 把键值对存入到map中
  b["Go语言"] = 2  //把key是Go语言的vlaue修改成2
  fmt.Printf("a的类型:%T,a的值:%v\n",a,a)
  fmt.Printf("b的类型:%T,b的值:%v\n",b,b)
  fmt.Printf("c的类型:%T,c的值:%v\n",c,c)
}

输出结果:

a的类型:[]int,a的值:[0 10 0 0 0]
b的类型:map[string]int,b的值:map[Go语言:2]
c的类型:[5]int,c的值:[0 10 0 0 0]


遍历比较

package main
import "fmt"
func main() {
  a := make([]int,3,10)
  b := make(map[string]int,10)
  c := [3]int{}
  a[0] =99
  b["Go语言"]= 99
  c[0] =99
  for k,v :=range c{
    fmt.Println(k,v)
  }
  for k,v :=range b {
    fmt.Println(k,v)
  }
  for k,v :=range a {
    fmt.Println(k,v)
  }
}

都可以使用for-range遍历

相关文章
|
11天前
|
Go
Go 语言为什么不支持并发读写 map?
Go 语言为什么不支持并发读写 map?
|
12天前
|
Go
Go 1.21的新特性: 切片和映射
Go 1.21的新特性: 切片和映射
|
13天前
|
算法 安全 Go
go 语言中 map 的相关知识
go 语言中 map 的相关知识
|
4天前
|
监控 Serverless Go
Golang 开发函数计算问题之Go 语言中切片扩容时需要拷贝原数组中的数据如何解决
Golang 开发函数计算问题之Go 语言中切片扩容时需要拷贝原数组中的数据如何解决
|
5天前
|
人工智能 Go
go切片参数传递用值还是指针
go切片参数传递用值还是指针
11 0
|
13天前
|
存储 Go 开发者
在Go中对切片/数组求和的多种方法
在Go中对切片/数组求和的多种方法
|
11月前
|
Go 索引
Go语言中的数组、切片和映射解析
Go语言中的数组、切片和映射解析
|
2月前
|
Go
go语言数组与切片
go语言数组与切片
30 0
|
2月前
|
JSON Go 数据格式
Go 语言基础之指针、复合类型【数组、切片、指针、map、struct】(4)
Go 语言基础之指针、复合类型【数组、切片、指针、map、struct】
|
2月前
|
Java 编译器 Go
Go 语言基础之指针、复合类型【数组、切片、指针、map、struct】(3)
Go 语言基础之指针、复合类型【数组、切片、指针、map、struct】