数组,可以存放多个相同类型数据的一种特殊的数据类型
一、基本语法及案例演示
语法格式
var 变量名称 [空间大小]数据类型
//我们在创建数组的时候会指定这个数据的空间有多大
//如:
//var hens [6]float64
//定义一个hens的数组, 空间大小位6,允许存放6个float64的值
案例
假设有一个养鸡场有6只鸡,他们的体重分别是 3kg,5kg,1kg,3.4kg,2kg,50kg
那这6只鸡 的总体重是多少?平均体重是多少?
package main
import (
"fmt"
)
func main(){
var hens [6]float64 //定义一个数组,空间大小为6
hens[0] = 3.0 //和其他语法的列表相似,hens[索引位],空间大小是6
hens[1] = 5.0 //索引位是0-5
hens[2] = 1.0
hens[3] = 3.4
hens[4] = 2.0
hens[5] = 50.0
totalWeight2 := 0.0 //定义变量接收总体重
for i := 0; i < len(hens); i++ { //循环遍历 将所有体重相加
totalWeight2 += hens[i]
}
avgWeight2 := fmt.Sprintf("%.2f", totalWeight2 / len(hens))
//fmt.Sprintf 将计算的值接收
//"%.2f"中的2表示精度 保留小数点后两位
//len 获取数组的长度
fmt.Printf("totalWeight2=%v avgWeight2=%v",totalWeight2, avgWeight2)
}
1、 数组的默认值
和前面所有的数据类型一样,数组定义后也是有默认值的,数组内数据的默认值取决于值的类型
package main
import (
"fmt"
)
func main() {
var intArr [3]int //比如这里是一个int的数组,int类型变量的默认值是0
fmt.Println(intArr)
}
返回
[0 0 0]
2、 数组的地址可以通过数组名来获取
package main
import (
"fmt"
)
func main() {
var intArr [3]int
fmt.Printf("intArr的内存地址=%p",&intArr) //也是通过& 来获取内存地址
}
返回
intArr的内存地址=0xc000016120
3、数组的第一个元素的地址,就是数组的首地址
package main
import (
"fmt"
)
func main() {
var intArr [3]int
fmt.Printf("%p\n%p\n%p\n%p\n",
&intArr,
&intArr[0],
&intArr[1],
&intArr[2])
}
返回
0xc000016120
0xc000016120
0xc000016128
0xc000016130
说明
1、通过上面结果可知,数组的内存地址是连续性的
2、intArr数组名 和intArr[0] 第一个索引位的地址是一样的,也证明了数组的第一个元素的地址,就是数组的首地址
3 intArr[0] 到intArr[1]顺延的内存地址,就是在数组定义时所指定的数据类型(int)的大小(字节数)的值
int类型占用8个字节,通过上面的接结果可知:
0xc000016120
0xc000016120 //索引位0 和数组地址相同
0xc000016128 //索引位1 基于0顺延一个int类型的8字节
0xc000016130 //索引位2 基于1顺延一个int的字节,
//因为这里是16进制,所以结果是30
4、改变数组中的值是不会修改内存地址的
package main
import (
"fmt"
)
func main() {
var intArr [3]int
intArr[0] = 10
intArr[1] = 20
intArr[2] = 30
fmt.Println(intArr)
fmt.Printf("%p\n%p\n%p\n%p\n",
&intArr,
&intArr[0],
&intArr[1],
&intArr[2])
intArr[0] = 40
intArr[1] = 50
intArr[2] = 60
fmt.Println(intArr)
fmt.Printf("%p\n%p\n%p\n%p\n",
&intArr,
&intArr[0],
&intArr[1],
&intArr[2])
}
返回
[10 20 30]
0xc000016120
0xc000016120
0xc000016128
0xc000016130
[40 50 60]
0xc000016120
0xc000016120
0xc000016128
0xc000016130
5、从终端接收数值存储到数组中
package main
import (
"fmt"
)
func main() {
var score [5]float64
for i := 0; i < len(score); i++{
fmt.Printf("请输入第%d个元素的值",i+1)
fmt.Scanln(&score[i]) //Scanln接收用户输入
}
fmt.Println(score) //输出整个数组的值
for i := 0; i < len(score); i++{
fmt.Printf("score[%d]=%v\n",i,score[i]) //%d 表示十进制显示,%v表示直接输出
}
}
返回
请输入第1个元素的值1
请输入第2个元素的值2
请输入第3个元素的值3
请输入第4个元素的值4
请输入第5个元素的值5
[1 2 3 4 5]
score[0]=1
score[1]=2
score[2]=3
score[3]=4
score[4]=5
二、四种初始化数组的方法
1、在定义时直接初始化
package main
import (
"fmt"
)
func main() {
var numArr01 [3]int = [3]int{1, 2, 3} //在声明数组的时候直接赋值
fmt.Println("numArr01",numArr01)
}
//numArr01 [1 2 3]
2、可以不定义数组类型,go有自己的类型推导自动识别
package main
import (
"fmt"
)
func main() {
var numArr02 = [3]int{1, 2, 3}
fmt.Println("numArr02",numArr02)
}
3、不设置数组的大小,自动识别元素的数量
package main
import (
"fmt"
)
func main() {
var numArr03 = [...]int{1, 2, 3} //...表示自动识别
fmt.Println("numArr03",numArr03)
}
4、给指定索引位添加元素
package main
import (
"fmt"
)
func main() {
//这里的1、0、2表示数组内的索引位
//这里是给指定的元素初始化值
var numArr04 = [...]int{1: 800, 0: 900, 2:999}
fmt.Println("numArr04",numArr04)
//类型推导也可以
numArr05 := [...]int{1: 800, 0: 900, 2:999}
fmt.Println("numArr05",numArr05)
}
//numArr04 [900 800 999]
//numArr05 [900 800 999]
三、遍历数组(for-range)
//格式
for index, value := range 数组列表 {
代码块
}
说明:
//index是数组循环获取的的下标,从0开始
//value"是数组循环获取的对应索引下标的值
//他们都是仅在for循环内部可见的局部变量
//遍历数组元素时,如果不需要使用下标index,可以直接把下班index标为下划线"_"
//index 和value名称不是固定的,可以自定义,一般命名为index和value
案例1 基本使用
package main
import (
"fmt"
)
func main() {
heroes := [...]string{"宋江","无用","卢俊义"}
for index, value := range heroes {
fmt.Printf("index=%v value=%v\n",index,value) //遍历输出索引位和对应的值
}
}
返回
index=0 value=宋江
index=1 value=无用
index=2 value=卢俊义
案例2 除去下标
package main
import (
"fmt"
)
func main() {
heroes := [...]string{"宋江","无用","卢俊义"}
for _, value := range heroes {
//如果不需要索引位的值,修改为下划线"_"即可
//将index改为下划线
//直接输入value就可以取值
fmt.Printf("value=%v\n",value)
}
}
返回
value=宋江
value=无用
value=卢俊义
四、数组使用细节和注意事项
1、数组不能动态变化
数组是多个"相同类型"数据的组合,一个数组一旦声明/定义了,长度是固定的,不能动态变化
案例
package main
import (
"fmt"
)
func main() {
var arr01 [3]int
arr01[0] = 1
arr01[1] = 30
arr01[2] = 1.1 //这里会报错,因为数组类型是int,而1.1是浮点值
arr01[3] = 890 //无效的 数组 索引 '3' (3 元素的数组超出界限)
//长度固定,不能动态变化,
}
2、设置数组时,如果不设置空间大小则视为切片
var arr []int //切片是啥后面再说
3、数组的类型可以是所有类型
数组中元素可以是任何类型,包括值类型和引用类型,但是不能混用
4、数组创建后,有默认值
数值类型数组 //0
字符串数组 //""
bool数组 //false
//等同于基本数据类型本身
案例
package main
import (
"fmt"
)
func main() {
//整数、浮点数、默认都为0
//字符串 默认 ""
//bool类型 默认是false
var arr01 [3]int
var arr02 [3]string
var arr03 [3]bool
fmt.Printf("arr01=%v\narr02=%v\narr03=%v\n",arr01,arr02,arr03)
}
返回
arr01=[0 0 0]
arr02=[ ]
arr03=[false false false]
5、 使用数组的步骤
1. 声明数组并开辟内存空间 //make和new
2. 给数组元素赋值(默认0值)
案例 new 和make的区别
6、数组的下标是从0开始的
package main
import (
"fmt"
)
func main() {
var arr04 [3]string //数组的下标是从0开始的 0-2
varr04[3] = "tom" //当索引位超出会报越界
}
7、go 的数组是值类型
package main
import (
"fmt"
)
func test01(arr [3]int){
arr[0] = 88 //值转递,相当于是将main的arr数组拷贝了一份
//所以在这里被修改后是不会影响到main中调用的数组的
}
func main() {
arr := [3]int{11,22,33}
test01(arr) //将数组作为参数传递
fmt.Println(arr)
}
返回
[11 22 33]
8、修改其他函数的数组
上面我们已经知道了数组是以值拷贝的形式进行传参的,但是我们想要其他函数中修改数组的值怎么做? 使用指针
案例
package main
import (
"fmt"
)
func test01(arr *[3]int){ //*开头表示 我这边是以指针的形式接收数据的
//引用指针,相当于直接调用main中arr变量的内存地址来修改数据
(*arr)[0] = 88 //在赋值时,我们去调用这个指针变量
//*arr去调用,但同时这是一个数组类型,所以要加括号
}
func main() {
arr := [3]int{11,22,33}
test01(&arr) //我们传参的时候将数组的内存地址进行传递
fmt.Println(arr)
}
返回
[88 22 33]
9、数组长度是数组类型的一部分
在传递函数参数时,需要考虑数组的长度. 判断以下三段代码是否正确
案例1
package main
import (
"fmt"
)
func modify(arr []int){ //这里接收参数时没有设置数组的长度,说明接收的就变成了切片
//正确的写法是arr [3]int
arr[0] = 100
fmt.Println("modify 的arr",arr)
}
func main(){
var arr = [...]int{1,2,3}
modify(arr)
}
案例2
package main
import (
"fmt"
)
func modify(arr [4]int){ //arr [4]int 和下面数组的长度不同
//下面的数组只有3位的长度,不能传递到4长度的数组
//不同长度的数组相当于是不同的数据类型
arr[0] = 100
fmt.Println("modify 的arr",arr)
}
func main(){
var arr = [...]int{1,2,3}
modify(arr)
}
案例3
package main
import (
"fmt"
)
func modify(arr [3]int){
arr[0] = 100
fmt.Println("modify 的arr",arr)
}
func main(){
var arr = [...]int{1,2,3}
modify(arr)
}
//正确
数组练习题
1、创建一个byte类型的26个元素的数组
创建一个byte类型的26个元素的数组,分别放置'A'-'Z' ,使用for循环访问所有元素并打印出来。
提示:
字符类型(byte)运算 'A'+ 1 -->'B' 这里的1也是byte类型才可以。 然后通过printf %c输出字符
package main
import "fmt"
func main(){
var shuzu [26]byte
for key,_ := range shuzu{ //for-range遍历数组
shuzu[key] = 'A' + byte(key) //'A'加索引位的byte作为值赋予
}
fmt.Printf("%c ",shuzu) //以字符形式输出
}
返回
[A B C D E F G H I J K L M N O P Q R S T U V W X Y Z]
练习2 请求出一个数组的最大值,并得到对应的下标
/*
1. 声明一个数组 var intArr[5] = [...]int {1,-1,9,90,11}
2. 假定第一个元素就是最大值,下标就是0
3. 然后从第二个元素开始循环比较,如果发现有更大的,则交换
*/
package main
import "fmt"
func main(){
var test = [5]int{12,2,21,45,5}
maxzhi := test[0]
xiabiao := 0
//这里len去获取这个数组的长度,然后循环每次去判断是否小于这个长度
//这里因为要从索引位1开始,1-4的位置,总长度是5,所以必须是小于,不能是小于等于,不然长度不够了
for i := 1; i < len(test);i++{
if maxzhi < test[i]{
maxzhi = test[i]
xiabiao = i
}
}
fmt.Println("最大值",maxzhi)
fmt.Println("最大值下标",xiabiao)
}
练习3 请求出一个数组的和 与他的平均值 for-range
/*
请求出一个数组和 和平均值 for-range
思路
1. 就是声明一个数组
2. 求出和sum
3. 求出平均值
*/
package main
import (
"fmt"
)
func main(){
var intArr2 = [...]int {1,2,3,4,7}
sum := 0
for _, val := range intArr2 {
//_下划线 val,表示只要值,不要索引下标
sum += val
}
fmt.Printf("sum=%v 平均值=%v\n", sum, sum / len(intArr2))
//上面显示的是sum=17 平均值=3
//这里应该是有小数点的,但是因为int的缘故都抹掉了
//所以我们需要修改一下平均值的数据类型和除的个数的数据类型
fmt.Printf("sum=%v 平均值=%v", float64(sum), float64(sum) / float64(len(intArr2)))
}
数组的复杂使用反转
随机生成5个数,并将其反转输出
案例
/*
思路
1. 随机生成5个数,rand Intn()函数
2. 当我们得到随机数后,就放入到一个数组中
3. 反转打印,交换的次数
第一个和倒数第一个元素交换,第二个和倒数第二个交换
如下
[1 2 3 4 5] 反转为 [5 4 3 2 1 ]
或者说
[1 2 3 4 5 6 ] 反转为 [ 6 5 4 3 2 1]
他反转的概念可以理解为如下
[1 5]
[ 2 4]
[3]
我们是从两边开始替换,然后向中间靠拢的
*/
package main
import (
"fmt"
"math/rand"
)
func main(){
var intArr3 [5]int
for i := 0; i < len(intArr3); i++{
intArr3[i] = rand.Intn(100) //随机生成0-100的整数
}
fmt.Println("我是交换之前的值",intArr3)
temp :=0
for i := 0; i < len(intArr3) / 2; i++{
//考虑到要进行反转,需要先筛选出一半的值存下来 比如 1 2
//然后拿另一半的值,覆盖之前取出值的位置 4 5 --》 1 2
//最后再将之前取出的值写入到之后取出值的位置 1 2 --> 4 5
//以此来完成前后值的反转
temp = intArr3[len(intArr3) - 1 - i]
//len(intArr3) - 1 来获取真正的索引位, (len的值是统计长度,是从1开始的,所以要减一)
//第一次循环,数组会拿到索引总长度-i的值,也就是0-4的最大值4位置的值
//将最后的索引位值 放入到临时变量中
intArr3[len(intArr3) - 1 - i ] = intArr3[i]
//然后将数组中第一个值写入到数组中最大值4的位置
intArr3[i] = temp
//最后将之前拿到的最后一位索引位的值,再赋予到第一位的值的索引位,完成反转
}
fmt.Println("我是交换之后的值",intArr3)
}
返回
我是交换之前的值 [81 87 47 59 81]
我是交换之后的值 [81 59 47 87 81]