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遍历

相关文章
|
1月前
|
存储 Go 索引
go语言中数组和切片
go语言中数组和切片
42 7
|
1月前
|
存储 Go 索引
go语言中的数组(Array)
go语言中的数组(Array)
107 67
|
1月前
|
Go 索引
go语言for遍历数组或切片
go语言for遍历数组或切片
106 62
|
1月前
|
Go
go语言for遍历映射(map)
go语言for遍历映射(map)
39 12
|
2月前
|
存储 Go
go语言 遍历映射(map)
go语言 遍历映射(map)
43 2
|
2月前
|
Go 索引
go语言遍历数组和切片
go语言遍历数组和切片
23 2
|
14天前
|
存储 监控 算法
员工上网行为监控中的Go语言算法:布隆过滤器的应用
在信息化高速发展的时代,企业上网行为监管至关重要。布隆过滤器作为一种高效、节省空间的概率性数据结构,适用于大规模URL查询与匹配,是实现精准上网行为管理的理想选择。本文探讨了布隆过滤器的原理及其优缺点,并展示了如何使用Go语言实现该算法,以提升企业网络管理效率和安全性。尽管存在误报等局限性,但合理配置下,布隆过滤器为企业提供了经济有效的解决方案。
57 8
员工上网行为监控中的Go语言算法:布隆过滤器的应用
|
1月前
|
Go 开发工具
百炼-千问模型通过openai接口构建assistant 等 go语言
由于阿里百炼平台通义千问大模型没有完善的go语言兼容openapi示例,并且官方答复assistant是不兼容openapi sdk的。 实际使用中发现是能够支持的,所以自己写了一个demo test示例,给大家做一个参考。
|
1月前
|
程序员 Go
go语言中结构体(Struct)
go语言中结构体(Struct)
103 71
|
9天前
|
算法 安全 Go
Go 语言中实现 RSA 加解密、签名验证算法
随着互联网的发展,安全需求日益增长。非对称加密算法RSA成为密码学中的重要代表。本文介绍如何使用Go语言和[forgoer/openssl](https://github.com/forgoer/openssl)库简化RSA加解密操作,包括秘钥生成、加解密及签名验证。该库还支持AES、DES等常用算法,安装简便,代码示例清晰易懂。
41 12