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月前
|
存储 JavaScript Java
(Python基础)新时代语言!一起学习Python吧!(四):dict字典和set类型;切片类型、列表生成式;map和reduce迭代器;filter过滤函数、sorted排序函数;lambda函数
dict字典 Python内置了字典:dict的支持,dict全称dictionary,在其他语言中也称为map,使用键-值(key-value)存储,具有极快的查找速度。 我们可以通过声明JS对象一样的方式声明dict
137 1
|
4月前
|
存储 缓存 JavaScript
Set和Map有什么区别?
Set和Map有什么区别?
365 1
|
1月前
|
存储 Java Go
【Golang】(3)条件判断与循环?切片和数组的关系?映射表与Map?三组关系傻傻分不清?本文带你了解基本的复杂类型与执行判断语句
在Go中,条件控制语句总共有三种if、switch、select。循环只有for,不过for可以充当while使用。如果想要了解这些知识点,初学者进入文章中来感受吧!
105 1
|
4月前
|
存储 人工智能 安全
深入理解 go sync.Map - 基本原理
本文介绍了 Go 语言中 `map` 在并发使用时的常见问题及其解决方案,重点对比了 `sync.Mutex`、`sync.RWMutex` 和 `sync.Map` 的性能差异及适用场景。文章指出,普通 `map` 不支持并发读写,容易引发错误;而 `sync.Map` 通过原子操作和优化设计,在某些场景下能显著提升性能。同时详细讲解了 `sync.Map` 的基本用法及其适合的应用环境,如读多写少或不同 goroutine 操作不同键的场景。
197 1
|
5月前
|
Go
【LeetCode 热题100】DP 实战进阶:最长递增子序列、乘积最大子数组、分割等和子集(力扣300 / 152/ 416 )(Go语言版)
本文深入解析三道经典的动态规划问题:**最长递增子序列(LIS)**、**乘积最大子数组** 和 **分割等和子集**。 - **300. LIS** 通过 `dp[i]` 表示以第 `i` 个元素结尾的最长递增子序列长度,支持 O(n²) 动态规划与 O(n log n) 的二分优化。 - **152. 乘积最大子数组** 利用正负数特性,同时维护最大值与最小值的状态转移方程。 - **416. 分割等和子集** 转化为 0-1 背包问题,通过布尔型 DP 实现子集和判断。 总结对比了三题的状态定义与解法技巧,并延伸至相关变种问题,助你掌握动态规划的核心思想与灵活应用!
221 1
|
6月前
|
Go 索引
Go语言数组的定义与操作 - 《Go语言实战指南》
本文介绍了 Go 语言中的数组(Array)相关知识,包括定义、初始化方式(默认、显式、指定索引及自动推导长度)、访问与修改、遍历方法(for 循环和 for range)、值类型特性(复制行为)、多维数组支持以及其与切片的区别。数组是定长且同类型的集合,适合性能敏感场景,但实际开发中更常用动态的切片(slice)。
212 11
|
5月前
|
存储 编译器 容器
set、map、multiset、multimap的介绍及使用以及区别,注意事项
set是按照一定次序存储元素的容器,使用set的迭代器遍历set中的元素,可以得到有序序列。set当中存储元素的value都是唯一的,不可以重复,因此可以使用set进行去重。set默认是升序的,但是其内部默认不是按照大于比较,而是按照小于比较。set中的元素不能被修改,因为set在底层是用二叉搜索树来实现的,若是对二叉搜索树当中某个结点的值进行了修改,那么这棵树将不再是二叉搜索树。
239 0
|
8月前
|
JavaScript 前端开发 API
JavaScript中通过array.map()实现数据转换、创建派生数组、异步数据流处理、复杂API请求、DOM操作、搜索和过滤等,array.map()的使用详解(附实际应用代码)
array.map()可以用来数据转换、创建派生数组、应用函数、链式调用、异步数据流处理、复杂API请求梳理、提供DOM操作、用来搜索和过滤等,比for好用太多了,主要是写法简单,并且非常直观,并且能提升代码的可读性,也就提升了Long Term代码的可维护性。 只有锻炼思维才能可持续地解决问题,只有思维才是真正值得学习和分享的核心要素。如果这篇博客能给您带来一点帮助,麻烦您点个赞支持一下,还可以收藏起来以备不时之需,有疑问和错误欢迎在评论区指出~
|
7月前
|
JavaScript 前端开发 开发者
flat、flatmap与map的用法区别
本文介绍了 JavaScript 数组方法 `flat()`、`flatMap()` 和 `map()` 的用法及区别。`flat()` 可按指定深度递归展平数组,参数为深度,默认一层;`flatMap()` 结合了 `map()` 和 `flat()` 功能,返回一维数组,长度可能不同于原数组;而 `map()` 返回与原数组长度一致的新数组。通过多个代码示例展示了三者的功能和差异,帮助开发者更好地理解和使用这些方法。
761 0