golang学习笔记 ---slice
- 指向底层数组的指针
- slice目前使用到的底层数组的元素个数,即长度
- 底层数组的最大长度,即容量
因此当我们定义一个切片变量,s := make([]int, 5, 10),即为指向了一个最大长度为10的底层数组,目前切片s使用到的长度为5。
基于slice的定义,在使用slice时,有以下几点注意事项:
- 当从左边界有截断时,会改变新切片容量大小
- 左边界默认0,最小为0;右边界默认slice的长度,最大为slice的容量
- 当然,因为指向同一个底层数组,对新slice的操作会影响到原来的slice
package main import (
"fmt"
) func main() { a := make([]int, 5, 10) b := a[:3]
//b len: 3 cap: 10
fmt.Println("b-- ", "len:", len(b), " ,cap:", cap(b)) //当从左边界有截断时 会改变新切片容量大小
c := a[2:4]
//c len: 2 cap: 8
fmt.Println("c-- ", "len:", len(c), " ,cap:", cap(c)) //左边界默认0 最小为0 | 右边界默认slice的长度 最大为slice的容量
d := a[:cap(a)]
//d len: 10 cap: 10
fmt.Println("d-- ", "len:", len(d), " ,cap:", cap(d))
}
输出:
b-- len: 3 ,cap: 10
c-- len: 2 ,cap: 8
d-- len: 10 ,cap: 10
2.slice的赋值及函数间传递
a := []int{1, 2, 3, 4, 5}
b := a
package main import (
"fmt"
) func main() { a := []int{1, 2, 3, 4, 5}
b := a //对b进行操作会影响到a
b[0] = 10
// 10 2 3 4 5
fmt.Println(a)
}
输出:
[10 2 3 4 5]
如上所示,则a, b指向同一个底层数组,且长度及容量因素相同,对b进行的操作会影响到a。
切片作为函数参数示例:
package main import (
"fmt"
) func modifySlice(s []int) {
s[0] = 10
} func main() {
a := []int{1, 2, 3, 4, 5}
modifySlice(a)
//[10 2 3 4 5]
fmt.Println(a)
}
输出:
[10 2 3 4 5]
如上所示,将slice作为参数在函数间传递的时候是值传递,产生了一个新的slice,只不过新的slice仍然指向原来的底层数组,所以通过新的slice也能改变原来的slice的值
package main import (
"fmt"
) func modifySlice(s []int) {
s = []int{10, 20, 30, 40, 50}
s[0] = -1
}
func main() {
a := []int{1, 2, 3, 4, 5}
modifySlice(a)
//[1 2 3 4 5]
fmt.Println(a)
}
如上所示,在调用函数的内部,将s整体赋值一个新的slice,并不会改变a的值,因为modifySlice函数内对s重新的整体赋值,让s指向了一个新的底层数组,而不是传递进来之前的a指向的那个数组,之后s的任何操作都不会影响到a了。
3.slice的append操作
func append(s []T, x ...T) []T
package main import (
"fmt"
) func main() {
a := make([]int, 2, 4)
fmt.Println("initialize.....")
fmt.Println("a:", a, " &a[0]:", &a[0], " len:", len(a), " cap:", cap(a)) //通常append操作都是将返回值赋值给自己,
//此处为了方便说明,没有这样做 //改变b的前2个元素值 会影响到a
a[0] = 99
fmt.Println("赋值之后.....")
fmt.Println("a:", a, " &a[0]:", &a[0], " len:", len(a), " cap:", cap(a))
fmt.Println("Append之后.....")
b := append(a, 2)
b[0] = 1
fmt.Println("a:", a, " &a[0]:", &a[0], " len:", len(a), " cap:", cap(a))
fmt.Println("b:", b, " &b[0]:", &b[0], " len:", len(b), " cap:", cap(b)) a = append(a, 3)
fmt.Println("a Append之后.....")
fmt.Println("a:", a, " &a[0]:", &a[0], " len:", len(a), " cap:", cap(a))
fmt.Println("b:", b, " &b[0]:", &b[0], " len:", len(b), " cap:", cap(b)) a = append(a, 4)
a = append(a, 5) fmt.Println("a 再次Append之后.....")
//a 扩容了,指向不同的底层数组
fmt.Println("a:", a, " &a[0]:", &a[0], " len:", len(a), " cap:", cap(a))
//a的扩容没有影响到b,b还是指向原来的底层数组
fmt.Println("b:", b, " &b[0]:", &b[0], " len:", len(b), " cap:", cap(b))
}
输出:
initialize.....
a: [0 0] &a[0]: 0xc0000105a0 len: 2 cap: 4
赋值之后.....
a: [99 0] &a[0]: 0xc0000105a0 len: 2 cap: 4
Append之后.....
a: [1 0] &a[0]: 0xc0000105a0 len: 2 cap: 4
b: [1 0 2] &b[0]: 0xc0000105a0 len: 3 cap: 4
a Append之后.....
a: [1 0 3] &a[0]: 0xc0000105a0 len: 3 cap: 4
b: [1 0 3] &b[0]: 0xc0000105a0 len: 3 cap: 4
a 再次Append之后.....
a: [1 0 3 4 5] &a[0]: 0xc000014240 len: 5 cap: 8
b: [1 0 3] &b[0]: 0xc0000105a0 len: 3 cap: 4
a 再次Append之后,a 扩容了,指向不同的底层数组,a的扩容没有影响到b,b还是指向原来的底层数组
package main import (
"fmt"
) func main() {
a := make([]int, 2, 4)
a[0] = 10
a[1] = 20
//a: [10 20] &a[0]: 0x10410020 len: 2 cap: 4
fmt.Println("a:", a, " &a[0]:", &a[0], " len:", len(a), " cap:", cap(a)) //进行append操作
b := append(a[:1], 1) //a: [10 1] &a[0]: 0x10410020 len: 2 cap: 4
fmt.Println("a:", a, " &a[0]:", &a[0], " len:", len(a), " cap:", cap(a)) //b: [10 1] &b[0]: 0x10410020 len: 2 cap: 4
fmt.Println("b:", b, " &b[0]:", &b[0], " len:", len(b), " cap:", cap(b))
}
输出:
a: [10 20] &a[0]: 0xc0000105a0 len: 2 cap: 4
a: [10 1] &a[0]: 0xc0000105a0 len: 2 cap: 4
b: [10 1] &b[0]: 0xc0000105a0 len: 2 cap: 4
package main import (
"fmt"
) func main() {
a := make([]int, 2, 4)
a[0] = 10
a[1] = 20
//a: [10 20] &a[0]: 0x10410020 len: 2 cap: 4
fmt.Println("a:", a, " &a[0]:", &a[0], " len:", len(a), " cap:", cap(a)) //进行append操作
//b := append(a[:1], 1)
c := a[:1]
fmt.Println("c:", c, " &c[0]:", &c[0], " len:", len(c), " cap:", cap(c))
b := append(c, 1) //a: [10 1] &a[0]: 0x10410020 len: 2 cap: 4
fmt.Println("a:", a, " &a[0]:", &a[0], " len:", len(a), " cap:", cap(a)) //b: [10 1] &b[0]: 0x10410020 len: 2 cap: 4
fmt.Println("b:", b, " &b[0]:", &b[0], " len:", len(b), " cap:", cap(b))
}
输出:
a: [10 20] &a[0]: 0xc000072140 len: 2 cap: 4
c: [10] &c[0]: 0xc000072140 len: 1 cap: 4
a: [10 1] &a[0]: 0xc000072140 len: 2 cap: 4
b: [10 1] &b[0]: 0xc000072140 len: 2 cap: 4
package main import (
"fmt"
) func main() { a := make([]int, 2, 4)
a[0] = 10
a[1] = 20
//a: [10 20] &a[0]: 0x10410020 len: 2 cap: 4
fmt.Println("a:", a, " &a[0]:", &a[0], " len:", len(a), " cap:", cap(a)) //进行append操作
//append是在第一个元素后开始追加,所以要超过容量,至少要追加4个,而不是之前例子的3个
b := append(a[:1], 1, 2, 3, 4) //a: [10 20] &a[0]: 0x10410020 len: 2 cap: 4
fmt.Println("a:", a, " &a[0]:", &a[0], " len:", len(a), " cap:", cap(a)) //b: [10 1 2 3 4] &b[0]: 0x10454020 len: 5 cap: 8
fmt.Println("b:", b, " &b[0]:", &b[0], " len:", len(b), " cap:", cap(b))
}
如果append的元素数较多,超过了原来的容量,直接采用了新的底层数组,也就不会影响到a了。
输出:
a: [10 20] &a[0]: 0xc0000105a0 len: 2 cap: 4
a: [10 20] &a[0]: 0xc0000105a0 len: 2 cap: 4
b: [10 1 2 3 4] &b[0]: 0xc000014240 len: 5 cap: 8
package main import (
"fmt"
) func testAppend(s []int) {
//进行append操作
s = append(s, 1)
//s: [10 1] &s[0]: 0x10410020 len: 2 cap: 4
fmt.Println("s:", s, " &s[0]:", &s[0], " len:", len(s), " cap:", cap(s))
}
func main() { a := make([]int, 2, 4)
a[0] = 10
a[1] = 20
//a: [10 20] &a[0]: 0x10410020 len: 2 cap: 4
fmt.Println("a:", a, " &a[0]:", &a[0], " len:", len(a), " cap:", cap(a)) testAppend(a[:1]) //a: [10 1] &a[0]: 0x10410020 len: 2 cap: 4
fmt.Println("a:", a, " &a[0]:", &a[0], " len:", len(a), " cap:", cap(a)) }
输出:
a: [10 20] &a[0]: 0xc0000105a0 len: 2 cap: 4
s: [10 1] &s[0]: 0xc0000105a0 len: 2 cap: 4
a: [10 1] &a[0]: 0xc0000105a0 len: 2 cap: 4
import (
"fmt"
) func testAppend(s []int) {
//进行append操作
s = append(s, 1, 2, 3, 4)
//s: [10 1] &s[0]: 0x10410020 len: 2 cap: 4
fmt.Println("s:", s, " &s[0]:", &s[0], " len:", len(s), " cap:", cap(s))
}
func main() { a := make([]int, 2, 4)
a[0] = 10
a[1] = 20
//a: [10 20] &a[0]: 0x10410020 len: 2 cap: 4
fmt.Println("a:", a, " &a[0]:", &a[0], " len:", len(a), " cap:", cap(a)) testAppend(a[:1]) //a: [10 1] &a[0]: 0x10410020 len: 2 cap: 4
fmt.Println("a:", a, " &a[0]:", &a[0], " len:", len(a), " cap:", cap(a)) }
输出
a: [10 20] &a[0]: 0xc0000105a0 len: 2 cap: 4 s: [10 1 2 3 4] &s[0]: 0xc000014240 len: 5 cap: 8 a: [10 20] &a[0]: 0xc0000105a0 len: 2 cap: 4
参考:
golang学习笔记 ---slice的更多相关文章
- golang学习笔记20 一道考察对并发多协程操作一个共享变量的面试题
golang学习笔记20 一道考察对并发多协程操作一个共享变量的面试题 下面这个程序运行的能num结果是什么? package main import ( "fmt" " ...
- golang学习笔记19 用Golang实现以太坊代币转账
golang学习笔记19 用Golang实现以太坊代币转账 在以太坊区块链中,我们称代币为Token,是以太坊区块链中每个人都可以任意发行的数字资产.并且它必须是遵循erc20标准的,至于erc20标 ...
- golang学习笔记18 用go语言编写移动端sdk和app开发gomobile
golang学习笔记18 用go语言编写移动端sdk和app开发gomobile gomobile的使用-用go语言编写移动端sdk和app开发https://blog.csdn.net/u01249 ...
- golang学习笔记17 爬虫技术路线图,python,java,nodejs,go语言,scrapy主流框架介绍
golang学习笔记17 爬虫技术路线图,python,java,nodejs,go语言,scrapy主流框架介绍 go语言爬虫框架:gocolly/colly,goquery,colly,chrom ...
- golang学习笔记16 beego orm 数据库操作
golang学习笔记16 beego orm 数据库操作 beego ORM 是一个强大的 Go 语言 ORM 框架.她的灵感主要来自 Django ORM 和 SQLAlchemy. 目前该框架仍处 ...
- golang学习笔记14 golang substring 截取字符串
golang学习笔记14 golang substring 截取字符串golang 没有java那样的substring函数,但支持直接根据 index 截取字符串mystr := "hel ...
- golang学习笔记13 Golang 类型转换整理 go语言string、int、int64、float64、complex 互相转换
golang学习笔记13 Golang 类型转换整理 go语言string.int.int64.float64.complex 互相转换 #string到intint,err:=strconv.Ato ...
- golang学习笔记12 beego table name `xxx` repeat register, must be unique 错误问题
golang学习笔记12 beego table name `xxx` repeat register, must be unique 错误问题 今天测试了重新建一个项目生成新的表,然后复制到旧的项目 ...
- golang学习笔记11 golang要用jetbrain的golang这个IDE工具开发才好
golang学习笔记11 golang要用jetbrain的golang这个IDE工具开发才好 jetbrain家的全套ide都很好用,一定要dark背景风格才装B 从File-->s ...
随机推荐
- django urls 配置小记
django urls 配置小记 首先应了解 django2.0在url的配置上较之以前的版本有点区别,在之前的版本是通过django.conf.urls.url函数来实现路径配置的 urlpatte ...
- nginx+tomcat集群时,tomcat参数优化
maxKeepAliveRequests=“1”: nginx动态的转给tomcat,nginx是不能keepalive的,而tomcat端默认开启了keepalive,会等待keepalive的ti ...
- Windows Server2008 R2 服务器域名设置Https安全证书访问
域名支持Https访问设置 1.首先登陆域名申办公司的域名管理账号添加TXT域名解析信息 以新网域名公司为例:http://dcp.xinnet.com,输入域名:www.xxx.com和密码登录即可 ...
- c# 第14节 字符方法、转义字符、字符串的方法
本节内容: 1:字符的定义 2:字符的方法 3: 转义字符 4:字符串简介 5:字符串方法 1:字符的定义 char与Unicode一一对应,一个char 2个字节. 2:字符的使用方法: 实例: s ...
- python27期尚哥讲并发编程:
python27day23并发编程----------------------------------------------------------------------------------- ...
- JUnit 4 和 TestNG
JUnit 4和TestNG都是Java中非常受欢迎的单元测试框架.两种框架在功能上看起来非常相似.哪一个更好?在Java项目中应该使用哪个单元测试框架? 下面表中概括了JUnit 4和TestNG之 ...
- 处理海量数据的grep、cut、awk、sed 命令
grep.cut.awk.sed 常常应用在查找日志.数据.输出结果等等,并对我们想要的数据进行提取. 通常grep,sed命令是对行进行提取,cut跟awk是对列进行提取 处理海量数据之grep命令 ...
- USACO wormhole
洛谷 P1444 [USACO1.3]虫洞wormhole https://www.luogu.org/problemnew/show/P1444 JDOJ 2386: USACO 2013 Dec ...
- sqler 2.2 发布了,支持定时任务以及触发器
sqler 在10前发布了,2.2 添加了定时任务以及触发器(webhook),都是比较方便的功能, 同时我也修改了dockerfile, 做了构建,添加了功能支持,同时push 到了dockerhu ...
- Debian 9 部分快捷键失效问题
教程 具体修复过程: 安装gnome-screensaver包,重启恢复正常.