Golang: 数组和切片
数组
同其他语言一样,数组是一些相同类型的元素的集合。数组声明
数组的类型为 n[T],其中 n 表示数组中元素的个数,T 表示数组中元素的类型。数组元素的个数 n 也是数组类型的一部分
package main
import "fmt"
func main() {
var a []int //int array a and length is 5
fmt.Println(a)
}
var a [5]int 是我们常用的一种声明数组的方式。同时也会给数组种的所有元素赋值int的初始值0.运行代码我们得到:[0 0 0 0 0] 我们还可以用shorhand 速记声明的方式声明和赋值数组:
package main
import "fmt"
func main() {
a := []int{, , , , } //int array a and length is 5
fmt.Println(a)
}
这个代码的运行结果是:[1 3 4 5 6]
如果没有给所有的元素赋值,那么后面的元素会被赋予默认值。
在声明数组的时候,可以使用...来代替数组长度,编译器会帮我们计算数组长度
package main
import "fmt"
func main() {
a := [...]int{, , } //int array a and length is 3
fmt.Println(len(a))
}
这个代码的运行结果是3,表明编译器帮我们计算好了数组长度。
由于数组的长度属于数组类型的一部分,那么[3]int和[5]int 不是相同一类型,不可以相互赋值。
数组遍历
Go提供了for rang来让我们遍历数组,这样比用for 循环简洁一些。
package main
import "fmt"
func main() {
a := [...]string{"a", "b"}
for i, v := range a { //range returns both the index and value
fmt.Printf(" the element %d of a is %s \n", i, v)
}
}
上面的代码运行结果是:
the element 0 of a is a
the element 1 of a is b
切片
由于数组长度不可变,导致数组的应用场景大幅减少。还好Go给我们提供了切片。
把上面的代码稍微修改一下,我们来创建一个切片
package main
import "fmt"
func main() {
a := [...]string{"a", "b", "c", "d", "e"}
for i, v := range a { //range returns both the index and value
fmt.Printf(" the element %d of a is %s \n", i, v)
}
b := a[:]
fmt.Println(b)
}
运行结果是
the element of a is a
the element of a is b
the element of a is c
the element of a is d
the element of a is e
[c]
我们通过 a[start:end] 这样的语法创建了一个从 a[start] 到 a[end -1] 的切片。在上面代码中,我们用这种方式创建了长度为1 的切片b。
我们还可以使用和创建数组相似的方式来创建切片
package main
import "fmt"
func main() {
b := []string{"a", "b", "c", "d", "e"}
fmt.Println(b)
}
需要注意的是[]中间是没有长度或者...的。这段代码创建了一个切片b,然后代码的运行结果是:[a b c d e]
切片的长度和容量
切片的长度就是切片中元素的个数。切片的容量是 切片创建的数组的长度减去创建切片时的start值。
具体我们看代码
package main
import "fmt"
func main() {
a := [...]string{"a", "b", "c", "d", "e"}
b := a[:]
fmt.Printf("length of slice %d capacity %d", len(b), cap(b)) //length is 1 and capacity is 4
}
运行结果是:length of slice 1 capacity 4
append 追加数据到切片上
package main
import "fmt"
func main() {
a := [...]string{"a", "b", "c", "d", "e"}
b := a[:]
fmt.Printf("length of slice %d capacity %d\n", len(b), cap(b)) //length is 1 and capacity is 4
b = append(b, "x", "y", "z", "m", "n")
fmt.Printf("length of slice %d capacity %d\n", len(b), cap(b)) //length is 1 and capacity is 8
}
第8行代码给切片b追加了5个元素,那么长度就是6,长度超过了容量4,go是怎么做的?如果容量超了,go会创建一个新的数组,容量翻倍,然后返回新数组的完整切片。
代码的运行结果是
length of slice capacity
length of slice capacity
我们还可以使用append把一个切片完整的追加到一个切片上:
package main
import "fmt"
func main() {
a := []string{"a", "b"}
b := []string{"c", "d", "e"}
a = append(a, b...)
fmt.Println(a)
}
运行结果是:
[a b c d e]
这里有点类似ES6的写法
var arr = [1, 2, 3]; var arr1 = [...arr, 4, 5, 6];
切片是数组的一种外层表示,所以我们每次修改切片,最终都会将修改落到数组上。
但是这时候再修改切边,修改就不会再反应到原来的数组上。
make创建切片
我们还可以使用函数make来创建切片。func make([]T, len, cap) []T,其中cap参数是可选的。
package main
import "fmt"
func main() {
a := make([]int, )
fmt.Println(a)
}
切片和数组作为参数传递
先上两段代码,第一段:
package main
import "fmt"
func main() {
a := []string{"a", "b", "c", "d", "e"}
fmt.Println("before call function:", a)
changeZeroElementA(a)
fmt.Println("after call function:", a)
b := []string{"a", "b", "c", "d", "e"}
fmt.Println("before call function:", b)
changeZeroElementB(b)
fmt.Println("after call function:", b)
}
func changeZeroElementA(arr []string) {
//arr = []string{"X", "Y"}
arr[] = "changed"
}
func changeZeroElementB(arr []string) {
arr[] = "changed"
}
结果是:
before call function: [a b c d e]
after call function: [changed b c d e]
before call function: [a b c d e]
after call function: [a b c d e]
第二段:
package main
import "fmt"
func main() {
a := []string{"a", "b", "c", "d", "e"}
fmt.Println("before call function:", a)
changeZeroElementA(a)
fmt.Println("after call function:", a)
b := []string{"a", "b", "c", "d", "e"}
fmt.Println("before call function:", b)
changeZeroElementB(b)
fmt.Println("after call function:", b)
}
func changeZeroElementA(arr []string) {
arr = []string{"X", "Y"}
arr[] = "changed"
}
func changeZeroElementB(arr []string) {
arr[] = "changed"
}
运行结果:
before call function: [a b c d e]
after call function: [a b c d e]
before call function: [a b c d e]
after call function: [a b c d e]
看到这里是不是有些奇怪,其实go里面一切都是值传递。这里有全方面的解释:https://stackoverflow.com/questions/39993688/are-golang-slices-pass-by-value
切片是数组的一种外层表示,所以我们每次修改切片,最终都会将修改落到数组上。
Golang: 数组和切片的更多相关文章
- Golang 数组、切片、映射
定义数组 var arr1 [5]int //整型类型 fmt.Println(arr1) //[0 0 0 0 0] //赋值 arr1 = [5]int{1, 2, 3, 4, 5} fmt.Pr ...
- golang数组与切片
golang中坑的慢慢踩! golang中的数组是值类型,函数调用是传入的是数组的拷贝,如果想改变数组的值,可考虑使用指针数组,即函数调用时传入数组的地址 golang中的切片是引用类型,但是在函数中 ...
- golang 数组、切片、map
一.数组(类似python的list) 数组的长度一旦定义了就不能动态增长.并且存储的数据类型必须相同. 创建方法: var 数组名 [长度]数据类型 例如: package main import ...
- Golang数组和切片的区别
大纲 数组是固定大小 切片不是动态数组,可以扩容 区别 定义方式不一样 初始化方法不一样 package main import "fmt" func main() { // -- ...
- golang学习笔记 ---数组与切片
数组: golang数组包含的每个数据称为数组元素(element),数组包含的元素个数被称为数组长度(length). golang数组的长度在定义后不可更改,并且在声明时可以是一个常量或常量表达式 ...
- golang 数组以及slice切片
老虞学GoLang笔记-数组和切片 数组 Arrays 数组是内置(build-in)类型,是一组同类型数据的集合,它是值类型,通过从0开始的下标索引访问元素值.在初始化后长度是固定的,无法修改其 ...
- 手把手golang教程【二】——数组与切片
本文始发于个人公众号:TechFlow,原创不易,求个关注 今天是golang专题的第五篇,这一篇我们将会了解golang中的数组和切片的使用. 数组与切片 golang当中数组和C++中的定义类似, ...
- golang笔记——数组与切片
一.切片的定义 我们可以从数组(go语言中很少直接使用数组)或者切片来初始化一个新的切片,也可以直接通过 make 来初始化一个所有元素为默认零值的切片. //1.通过数组来初始化切片 arr := ...
- [Golang学习笔记] 07 数组和切片
01-06回顾: Go语言开发环境配置, 常用源码文件写法, 程序实体(尤其是变量)及其相关各种概念和编程技巧: 类型推断,变量重声明,可重名变量,类型推断,类型转换,别名类型和潜在类型 数组: 数组 ...
随机推荐
- unity的一些tips
主要是我知乎上回答的一个关于unity的tip,备忘. 说说我所看到unity相关的,不好的习惯: 1 尽量不要在Awake(), start()等函数内加入业务逻辑的初始化代码.首先无法简便的直接启 ...
- 【java】之cron表达式
秒(~) 分钟(~) 小时(~) 天(月)(~,但是你需要考虑你月的天数) 月(~) 天(星期)(~ =SUN 或 SUN,MON,TUE,WED,THU,FRI,SAT) .年份(-) 其中每个元素 ...
- Windows不要使用记事本编辑文本文件
摘自:廖雪峰 千万不要使用Windows自带的记事本编辑任何文本文件.原因是Microsoft开发记事本的团队使用了一个非常弱智的行为来保存UTF-8编码的文件,他们自作聪明地在每个文件开头添加了0x ...
- 知识点:spring 完全手册
什么是spring spring是一个开源框架,为简化企业级开发而生,使用spring可以使简单的java bean 实现以前只有EJG才能实现的功能. Spring是一个轻量级的控制反转(IoC)和 ...
- Java执行JavaScript代码
Java执行JavaScript代码 这篇文章主要为大家详细介绍了Java执行JavaScript代码的具体操作方法,感兴趣的小伙伴们可以参考一下 我们要在Java中执行JavaScriptMetho ...
- .NET自动化测试工具:Selenium Grid
在生产环境,QA会同时跑几十个上百个的test case.如果用单机串行的话,是一件非常耗时的事情,估计比手点快不了多少.使用并行方案的话,有两种方法,一个是自己写并行框架,一个是用现成的Seleni ...
- js 一些方法
1.js去除字符串前后的空格 function Trim(str) { return str.replace(/(^\s*)|(\s*$)/g, ""); } 2.js打乱数组的顺 ...
- SpringMVC之DispatcherServlet类图
DispatcherServlet类图 在IntelliJ IDEA 中打开其源码,然后右键查看Diagrams,工具会自动生成其类图. 可是直观的看出,DispatcherServlet本质上是Se ...
- java中存在三种调用机制
1:同步调用:一种阻塞式调用,调用方要等待对方执行完毕才返回,它是一种单向调用 2:回调:一种双向调用模式,也就是说,被调用方在接口被调用时也会调用对方的接口: 3:异步调用:一种类似消息或事件的机制 ...
- rpc调用过程
在openstack中,各个组件之间的调用遵循RESTful风格,而组件内部各服务之间的相互调用采用rpc远程调用,比如nova-conductor和nova-compute rpc原理: 首先了解什 ...