go特性-数组与切片
数组:
复制传递(不要按照c/c++的方式去理解,c/c++中数组是引用传递),定长
切片:
引用传递,底层实现是3个字段 array(数组) + len(长度) +cap(容量)
type slice struct {
array unsafe.Pointer
len int
cap int
}
要特别注意的是,切片的引用传递指的是切片传递时,切片的array字段是引用传递的,len和cap字段依然是赋值传递。
写个伪代码:
Array []interface{}
Len int
Cap int
}
func fake() {
slice1 := &Slice{
Array: 指向一块连续内存的定长数组,
Len: 0,
Cap: 4
}
slice2 := slice1 // 这一行相当于如下:
slice2 := &Slice{}
slice2.Array = slice1.Array // 指向同一块内存,所以说切片是引用传递
slice2.Len = slice1.Len // len字段被复制了
slice2.Cap = slice2.Cap // cap字段被复制了
// 后续如果触发了任一slice的array重新分配内存,另一个slice都是不知道的
// 已经修改任一slice的len和cap,另一个slice也是不知道的
}
最后总结题:
/*
输出结果:
array1: [1 2 3] array2 [100 2 3]
slice1: [100 2 3] slice2 [100 2 3]
*/
func TestSlice1(t *testing.T) {
// 数组是赋值传递
array1 := [3]int{1,2,3}
// 这里array1复制了一份,array2和array1已经不是同一份数据了
// 所以对数组array1,array2的修改是互不影响的
array2 := array1
array2[0] = 100
fmt.Println("array1:", array1, "array2", array2)
// 切片是引用传递
slice1 := make([]int, 0)
slice1 = append(slice1, 1, 2, 3)
// slice2和slice1引用的同一份数据,所以slice2对已有元素的修改,会影响到slice1
// 这里有个细节需要注意,往下面TestSlice2继续看
slice2 := slice1
slice2[0] = 100
fmt.Println("slice1:", slice1, "slice2", slice2)
}
/*
输出结果
slice1: [1 2 3] slice2 [1 2 3 4]
slice1: [1 2 3] slice2 [1 2 3 4 5]
slice1: [1 2 3] slice2 [100 2 3 4 5]
*/
func TestSlice2(t *testing.T) {
// 切片是引用传递
slice1 := make([]int, 0, 4) // 容量是4
slice1 = append(slice1, 1, 2, 3)
// slice2和slice1引用的同一份数据,所以slice2对元素的修改,会影响到slice1
// 特别要注意的是,这里说的引用同一份数据,实际上是指的slice1和slice2内部的ptr指向了同一个数组
// 但是slice1和slice2结构中的len和cap是复制传递的
slice2 := slice1
// 添加第4个元素,容量足够,不会触发内部ptr数组重新分配
slice2 = append(slice2, 4)
fmt.Println("slice1:", slice1, "slice2", slice2)
// 添加第5个元素,容量不足,slice2.ptr重新分配内存,此时slice2.ptr和slice1.ptr已经不是同一份内存了
slice2 = append(slice2, 5)
fmt.Println("slice1:", slice1, "slice2", slice2)
// 所以这里slice2对已有元素的修改,不会影响到slice1了
slice2[0] = 100
fmt.Println("slice1:", slice1, "slice2", slice2)
}
```**------------恢复内容开始------------**
**数组:**
复制传递(不要按照c/c++的方式去理解,c/c++中数组是引用传递),定长
**切片:**
引用传递,底层实现是3个字段 array(数组) + len(长度) +cap(容量)
```go/src/runtime/slice.go slice结构定义:
type slice struct {
array unsafe.Pointer
len int
cap int
}
要特别注意的是,切片的引用传递指的是切片传递时,切片的array字段是引用传递的,len和cap字段依然是赋值传递。
写个伪代码:
Array []interface{}
Len int
Cap int
}
func fake() {
slice1 := &Slice{
Array: 指向一块连续内存的定长数组,
Len: 0,
Cap: 4
}
slice2 := slice1 // 这一行相当于如下:
slice2 := &Slice{}
slice2.Array = slice1.Array // 指向同一块内存,所以说切片是引用传递
slice2.Len = slice1.Len // len字段被复制了
slice2.Cap = slice2.Cap // cap字段被复制了
// 后续如果触发了任一slice的array重新分配内存,另一个slice都是不知道的
// 已经修改任一slice的len和cap,另一个slice也是不知道的
}
最后总结题:
/*
输出结果:
array1: [1 2 3] array2 [100 2 3]
slice1: [100 2 3] slice2 [100 2 3]
*/
func TestSlice1(t *testing.T) {
// 数组是赋值传递
array1 := [3]int{1,2,3}
// 这里array1复制了一份,array2和array1已经不是同一份数据了
// 所以对数组array1,array2的修改是互不影响的
array2 := array1
array2[0] = 100
fmt.Println("array1:", array1, "array2", array2)
// 切片是引用传递
slice1 := make([]int, 0)
slice1 = append(slice1, 1, 2, 3)
// slice2和slice1引用的同一份数据,所以slice2对已有元素的修改,会影响到slice1
// 这里有个细节需要注意,往下面TestSlice2继续看
slice2 := slice1
slice2[0] = 100
fmt.Println("slice1:", slice1, "slice2", slice2)
}
/*
输出结果
slice1: [1 2 3] slice2 [1 2 3 4]
slice1: [1 2 3] slice2 [1 2 3 4 5]
slice1: [1 2 3] slice2 [100 2 3 4 5]
*/
func TestSlice2(t *testing.T) {
// 切片是引用传递
slice1 := make([]int, 0, 4) // 容量是4
slice1 = append(slice1, 1, 2, 3)
// slice2和slice1引用的同一份数据,所以slice2对元素的修改,会影响到slice1
// 特别要注意的是,这里说的引用同一份数据,实际上是指的slice1和slice2内部的ptr指向了同一个数组
// 但是slice1和slice2结构中的len和cap是复制传递的
slice2 := slice1
// 添加第4个元素,容量足够,不会触发内部ptr数组重新分配
slice2 = append(slice2, 4)
fmt.Println("slice1:", slice1, "slice2", slice2)
// 添加第5个元素,容量不足,slice2.ptr重新分配内存,此时slice2.ptr和slice1.ptr已经不是同一份内存了
slice2 = append(slice2, 5)
fmt.Println("slice1:", slice1, "slice2", slice2)
// 所以这里slice2对已有元素的修改,不会影响到slice1了
slice2[0] = 100
fmt.Println("slice1:", slice1, "slice2", slice2)
}
------------恢复内容结束------------
go特性-数组与切片的更多相关文章
- Go语言--数组、切片、
3.1 数组--固定大小的连续空间 3.1.1 声明数组 写法 var 数组变量名 [元素数量]T 说明: 变量名就是使用时的变量 元素的数量可以是表达式,最后必须为整型数值 T 可是是任意基本类型, ...
- go语言之行--数组、切片、map
一.内置函数 append :追加元素到slice里,返回修改后的slice close :关闭channel delete :从map中删除key对应的value panic : 用于异常处理,停 ...
- 《Go语言实战》笔记之第四章 ----数组、切片、映射
原文地址: http://www.niu12.com/article/11 ####数组 数组是一个长度固定的数据类型,用于存储一段具有相同的类型的元素的连续块. 数组存储的类型可以是内置类型,如整型 ...
- Go语言入门——数组、切片和映射(下)
上篇主要介绍了Go语言里面常见的复合数据类型的声明和初始化. 这篇主要针对数组.切片和映射这些复合数据类型从其他几个方面介绍比较下. 1.遍历 不管是数组.切片还是映射结构,都是一种集合类型,要从这些 ...
- go 学习笔记之数组还是切片都没什么不一样
上篇文章中详细介绍了 Go 的基础语言,指出了 Go 和其他主流的编程语言的差异性,比较侧重于语法细节,相信只要稍加记忆就能轻松从已有的编程语言切换到 Go 语言的编程习惯中,尽管这种切换可能并不是特 ...
- PHP转Go系列:数组与切片
数组的定义 用过PHP的同学应该很清楚,无论多么复杂的数据格式都可以用数组来表达,什么类型的数据都可以往里塞,它是工作必备的一部分,使用很简单,易用程度简直变态. $array = [1, 'name ...
- go语言之字符串、指针、数组、切片、结构struct、面向对象
一: 字符串 概述: Go 语言将字符串作为 种原生的基本数据类型,字 符串的初始化可以使用字符串字面量. (1)字符串是常量,可以通过类 数组 索引访问其字节单元,但是不能修改某个字节的值 (2)宇 ...
- PHP转Go系列:数组与切片 转
数组的定义# 用过PHP的同学应该很清楚,无论多么复杂的数据格式都可以用数组来表达,什么类型的数据都可以往里塞,它是工作必备的一部分,使用很简单,易用程度简直变态. Copy $array = [1, ...
- GO语言总结(3)——数组和切片
上篇博文简单介绍了一下Go语言的基本类型——GO语言总结(2)——基本类型,本篇博文开始介绍Go语言的数组和切片. 一.数组 与其他大多数语言类似,Go语言的数组也是一个元素类型相同的定长的序列. ( ...
随机推荐
- Docker部署Mysql, Tomcat, Nginx, Redis
1. Mysql部署 问题及解决方案 容器内的网络和外部机器不能直接通信 外部机器和宿主机可以直接通信 宿主机和容器可以直接通信 当容器中的网络服务需要被外部机器访问时,可以将容器中提供服务的端口映射 ...
- Shell脚本实现---Swarm集群部署实例(Swarm Cluster)
Shell脚本实现---Swarm集群部署实例(Swarm Cluster) 一.机器环境(均是centos7.8) IP hostname 角色 192.168.10.200 manager-swa ...
- 一起学Vue:路由(vue-router)
前言 学习vue-router就要先了解路由是什么?前端路由的实现原理?vue-router如何使用?等等这些问题,就是本篇要探讨的主要问题. vue-router是什么 路由是什么? 大概有两种说法 ...
- Tomcat启动中文乱码
将Tomcat中conf里面的logging.properties修改一下编码方式: java.util.logging.ConsoleHandler.encoding = UTF-8 改成 java ...
- vue API 知识点(1)---全局 API 总结
1.Vue.extend(options) 构造器,创建一个 子类 .参数是一个包含组件选项的对象 data 选项是特例,需要注意 在 Vue.extend() 中它必须是一个函数, <div ...
- 「APIO2015」巴邻旁之桥 Palembang Bridges
贪心 先转化一下题意 首先如果一个人的家和办公室在河同一侧那么建桥的时候不用去考虑它,最终把答案加上即可 在河两侧的家和办公室互换不影响答案,那么可以把这个抽象到一个区间$[l,r]$,距离就是$|l ...
- 「BalticOI 2020」病毒
AC自动机+DP最短路转移 怎么说呢,挺套路的,也不是太难,但是一上手会被大量的信息淹没思路,还是要注意关注主要信息,不要被一些细节卡住 由于抗体是要在基因序里面出现过,那么考虑把抗体的序列检出AC自 ...
- .NET 5.0正式发布,功能特性介绍(翻译)
本文由葡萄城技术团队翻译并首发 转载请注明出处:葡萄城官网,葡萄城为开发者提供专业的开发工具.解决方案和服务,赋能开发者. 我们很高兴今天.NET5.0正式发布.这是一个重要的版本-其中也包括了C# ...
- Sublime Text 3 安装插件与快捷键总结
ublime Text 3 是一个了不起的软件.首先,它是一个干净,实用,可以快速的编写代码编辑器.它不仅具有令人难以置信的内置功能(多行编辑和VIM模式),而且还支持插件,代码片段和其他许多东西.很 ...
- javascript链式运动框架案例
javascript链式运动框架 任务描述: 当鼠标移入红色矩形时,该矩形宽度逐渐增加至400px,之后高度逐渐增加至400px; 当鼠标移出红色矩形时,该矩形高度逐渐减小至200px,之后宽度逐渐减 ...