go——数组(二)
1.内部实现
在Go语言里,数组是一个长度固定的数据类型,用于存储一段具有相同的类型的元素的连续块。
数组存储的类型可以是内置类型,如整型或字符串,也可以是某种结构类型。
灰格子代表数组里面的元素,每个元素都紧邻另一个元素。
每个元素都属于相同的类型,并且每个元素可以用一个唯一的索引(也称下标或标号)来访问。
数组是一种非常有用的数据结构,因为其占用的内存是连续分配的。
由于内存连续,CPU能把正在使用的数据缓存更久的时间。
而且内存连续很容易计算索引,可以快速迭代数组里的所有元素。
数组的类型信息可以提供每次访问一个元素时需要在内存中移动的距离。
既然数组的每个元素类型相同,又是连续分配,就可以以固定速度索引数组中的任意数据,速度非常快。
2.声明和初始化
声明数组时需要指定内部存储的数据类型,以及需要存储的元素的数量,这个数量也称为数组的长度。
(1)声明一个数组,并设置为零值
//声明一个包含5个元素的整型数组
var array [5]int
一旦声明,数组里存储的数据类型和长度就都不能改变了。
如果需要存储更多的元素,就需要先创建一个更长的数据,再把原来数组里的值复制到新数组里。
在Go语言中声明变量时,总会使用对应类型的零值来对变量进行初始化。数组也不例外。
当数组初始化时,数组内每个元素都初始化为对应类型的零值。
整型数组里的每个元素都初始化为0,也就是整数的零值。
(2)使用数组字面量声明数组
一种快速创建数组并初始化的方式是使用数组字面量。
数组字面量允许声明数组里元素的数量同时指定为每个元素的值。
//声明一个包含5个元素的整型数组
//用具体值初始化每个元素
array := [5]int{10,20,30,40,50}
(3)让Go自动计算声明数组的长度
如果使用...代替数组的长度,Go语言会根据初始化时数组元素的数量来确定该数组的长度。
//声明一个整型数组
//用具体值初始化每个元素
//容量由初始化值的数量决定
array := [...]int{10,20,30,40,50}
(4)声明数组并指定特定元素的值
如果知道数组的长度,而且准备给每个值都指定具体值,就可以通过索引来进行指定。
//声明一个有5个元素的数组
//用具体值初始化索引为1和2的元素
//其余元素保持零值
array := [5]int{1:20, 2: 30}
3.使用数组
(1)访问数组元素
因为内存布局是连续的,所以数组是效率很高的数据结构。
在访问数组里任意元素的时候,这种高效都是数组的优势。
要访问数组里某个单独元素,使用[]运算符。
//声明一个包含5个元素的整型数组
//用具体值初始为每个元素
array := [5]int{10,20,30,40,50} //修改索引为2的元素的值
array[2] = 35
(2)访问指针数组的元素
声明一个所有元素都是指针的数组。使用*运算符就可以访问元素指针所指向的值
//声明包含5个元素的指向整数的数组
//用整型指针初始化索引为0和1的数组元素
array := [5]*int(0: new(int), 1: new(int)) //为索引为0和1的元素赋值
*array[0] = 10
*array[1] = 20
(3)把同样类型的一个数组赋值给另外一个数组
在Go语言里,数组是一个值。这意味着数组可以用在赋值操作中。
变量名代表整个数组,因此,同样类型的数组可以赋值给另一个数组。
//声明第一个包含5个元素的字符串数组
var array1 [5]string //声明第二个包含5个元素的字符串数组
//用颜色初始化数组
array2 := [5]string{"red", "yellow", "blue", "green", "pink"} //把array2的值赋值给array1
array1 = array2
(4)编译器会阻止类型不同的数组互相赋值
数组变量的类型包括数组长度和每个元素的类型。
只有这两部分都相同的数组,才是类型相同的数组,才能相互赋值。
//声明第一个包含4个元素的字符串数组
var array1 [4]string //声明第二个包含5个元素的字符串数组
//初始化数组
array2 := [5]string{"red", "blue", "yellow", "green", "pink"} //复制
array1 = array2 //cannot use array2 (type [5]string) as type [4]string in assignment
(5)把一个指针数组赋值给另一个
复制指针数组,只会复制指针的值,而不会复制指针所指向的值。
//声明第一个包含3个元素的指向字符串的指针数组
var array1 [3]*string //声明第二个包含3个元素的指向字符串的指针数组
//使用字符串指针初始化这个数组
array2 := [3]*string{new(string), new(string), new(string)} //*array[]表示反取元素的值
//赋值
*array2[0] = "blue"
*array2[1] = "pink"
*array2[2] = "yellow" //复制
array1 = array2
4.多维数组
数组本身只有一个维度,不过可以组合多个数组创建多维数组。
多维数组很容器管理具有父子关系的数据或者与坐标系相关联的数据。
(1)声明二维数组
//声明一个二维整型数组,两个维度分别存储4个元素和2个元素
var array [4][2]int //使用数组字面量来声明并初始化一个二维整型数组
array := [4][2]int{{10, 11}, {20, 21}, {30, 31}, {40, 41} //声明并初始化外层索引为1和2的元素
array := [4][2]int{1: {20, 21}, 2: {30, 31}} //声明并初始化外层数组和内层数组的单个元素
array := [4][2]int{1: {0: 20}, 2: {1: 31}}
(2)访问二维数组的元素
为了访问单个元素,需要反复组合使用[]操作符。
//声明一个2X2的二维整型数组
var array [2][2]int //设置每个元素的整型值
array[0][0] = 10
array[0][1] = 20
array[1][0] = 30
array[1][1] = 40
(3)同样类型的多维数组赋值
只要类型一致就可以将多维数组相互赋值。
多维数组的类型包括每一维度的长度以及最终存储在元素中的数据类型。
//声明两个不同的二维整型数组
var array1 [2][2]int
var array2 [2][2]int //设置每个元素的整型值
array2[0][0] = 10
array2[0][1] = 20
array2[1][0] = 30
array2[1][1] = 40 //复制
array1 =array2
(4)使用索引为多维数组赋值
//将array1的索引为1的维度复制到一个同类型的新数组里面
var array3 [2]int = array1[1] //将外层数组的索引为1、内层数组的索引为0的整型值赋值到新的整型变量里
var value int = array1[1][0]
5.在函数间传递数组
根据内存和性能来看,在函数间传递数组是一个开销很多的操作。
在函数之间传递变量时,总是以值的变量传递。
如果这个变量是一个数组,意味着整个数组不管有多长,都会完整复制,并传递给函数。
假设现在我们有一个包含100万个int类型元素的数组。
在64位架构上,需要800万字节,也就是8MB内存。
(1)使用值传递在函数间传递大数组
//声明一个需要8MB的数组
var array [1000000]int //将数组传递给foo函数
foo(array) //函数foo接受一个100万个整型的数组
func foo(array [1000000]int) {
...
}
每次函数foo被调用时,必须在栈上分配8MB的内存。
之后整个数组的值被复制到刚刚分配的内存里。
虽然Go的垃圾回收机制还不错,但是如果大规模调用,势必会有很多资源消耗。
其实我们只需要传递指针,这样只需要8字节的内存分配给指针就可以了。
(2)使用指针在函数间传递大数组
//声明一个需要8MB的数组
var array [1000000]int //将数组传递给foo函数
foo(&array) //函数foo接受一个100万个整型的数组
func foo(array *[1000000]int) {
...
}
go——数组(二)的更多相关文章
- hiho一下121周 后缀数组二·重复旋律2
后缀数组二·重复旋律2 时间限制:5000ms 单点时限:1000ms 内存限制:256MB 描述 小Hi平时的一大兴趣爱好就是演奏钢琴.我们知道一个音乐旋律被表示为长度为 N 的数构成的数列.小Hi ...
- Java基本语法-----java数组(一维数组二维数组)
嘿嘿!你们懂的,又是图片,委屈大家了. java数组(一维数组二维数组) [正在看本人博客的这位童鞋,我看你气度不凡,谈吐间隐隐有王者之气,日后必有一番作为!下面有个"顶"字,你就 ...
- C++学习(十五)(C语言部分)之 数组二
数组大纲 多维数组 二维数组 重点 (三位以上基本不会用到) 都是用来存数据 一个班有20个人 可以用一维数组存20个人的年龄 int age[20]; 一个年级10个班 每个班20人 int age ...
- javascript数据结构和算法 第二章 (数组) 二
字符串表示的数组 join() 和 toString() 函数返回数组的字符串表示.这两个函数通过将数组中的元素用逗号分隔符切割,返回字符串数组表示. 这里有个样例: var names = [&qu ...
- hihocoder #1407 : 后缀数组二·重复旋律2
#1407 : 后缀数组二·重复旋律2 Time Limit:5000ms Case Time Limit:1000ms Memory Limit:256MB 描述 小Hi平时的一大兴趣爱好就是演奏钢 ...
- 数据结构 二维数组-->稀疏数组-->二维数组
稀疏数组基本概念: 稀疏数组应用场景: 当一个数组大部分的元素为"0",或者为同一个值的数组时,可以使用稀疏数组来保存该数组 处理方法: 1>记录数组一共有几行几列,有多少不 ...
- Java 一维数组 二维数组 三维数组
二维数组包含一位数组 三维数组就是在二维数组的基础上,再加一层.把二维数组看做是一维数组就可以了,按照上述理解类推. 下面是 一维 二维 三维数组例子 一维数组: int[] array1 ...
- java 获取数组(二维数组)长度实例程序
我们可能知道 js有个length函数,java也有啊length函数 例 如果数组是data[],则data.length 代码如下 复制代码 byte[] phone =new byte[81]; ...
- 【转】C++动态创建二维数组,二维数组指针
原作者博客:蒋国宝的IT技术博客 今天完成一道题目需要自己用指针创建一个二维的数组,不得不承认指针的确是恶心. int **result; ; ; result = new int*[row]; ; ...
- php php打乱数组二维数组、多维数组
php中的shuffle函数只能打乱一维数组,有什么办法快速便捷的打乱多维数组?手册上提供了 <?php function shuffle_assoc($list) { if (!is ...
随机推荐
- JQ 时间插件
<script type="text/javascript" charset="utf-8" src="__PUBLIC__/ueditor/u ...
- Java 设计模式01 - 简单工厂模式
先要学习设计模式之前的先看看一些基础 UML类图简单说明 可以先看看我的这篇博客: UML类图简单说明,学习编程思路的必会技能 接下来才是重点,开始我们的旅程吧. 一.UML类图展示 我们要用简单工厂 ...
- JSP页面之间传递参数的方法有哪些?
JSP页面之间传递参数的方法有哪些? 解答: 1)request 2)session 3)application 4)提交表单 5)超链接
- python socket 简单例子
myserver.py: import socket serversocket = socket.socket(socket.AF_INET, socket.SOCK_STREAM) serverso ...
- (转)java Exception层次结构详解
转自:http://www.importnew.com/14688.html 1. JAVA异常层次结构 异常指不期而至的各种状况,如:文件找不到.网络连接失败.非法参数等.异常是一个事件,它发生在程 ...
- phpStorm格式化代码快捷键
Ctrl+Alt+L
- SurvivalShooter学习笔记(三.敌人移动)
1.敌人和玩家若存活,敌人始终朝着玩家所在位置移动,所以要给玩家物体一个Tag:Player从而找到玩家 2.敌人的自动寻路使用Unity自带的NavMeshAgent寻路组件寻路,要先把场景中不动的 ...
- js高级面试题
<script> //1. var foo = { bar: function () { return this.baz; }, baz: 1 }; (function () { retu ...
- 【BZOJ1951】[Sdoi2010]古代猪文 Lucas定理+CRT
[BZOJ1951][Sdoi2010]古代猪文 Description 求$X=\sum\limits_{d|n}C_n^d$,$Ans=G^X (\mod 999911659)$. Input 有 ...
- QQ能上,网页打不开
这是一个老问题了,在大学的时候就经常碰到有人问这样的问题,今天写出来祭奠一下,姑凉长点心吧~! 安阳地区DNS:网通202.102.224.68 如果你是电信:222.88.88.88或者直接弄成顶级 ...