一道常见的 Go 面试题是:把切片传给函数,在函数里执行 append,调用方的数据会不会改变?
回答“切片是引用类型,所以会变”,或者“Go 只有值传递,所以不会变”,都不完整。这个细节在参数拼装、权限列表、查询过滤条件中经常造成偶发覆盖。通过几段短代码,就能把它变成开发中真正有用的小技巧。
先看一个最简单的函数,它试图给列表追加一个元素:
一道常见的 Go 面试题是:把切片传给函数,在函数里执行 append,调用方的数据会不会改变?
回答“切片是引用类型,所以会变”,或者“Go 只有值传递,所以不会变”,都不完整。这个细节在参数拼装、权限列表、查询过滤条件中经常造成偶发覆盖。通过几段短代码,就能把它变成开发中真正有用的小技巧。
先看一个最简单的函数,它试图给列表追加一个元素:
在日常开发中,我们常常需要处理动态数据集合。Go语言提供了多种数据结构,其中container/list包实现的双向链表和内置的切片(slice)是最常用的两种线性结构。但何时该选择链表而非切片呢?这篇文章分享一下我的理解。
切片是基于数组的动态序列,元素在内存中连续存储。这种结构使得随机访问效率极高(O(1)时间复杂度),但在中间位置插入或删除元素时需要移动后续所有元素,时间复杂度为O(n)。
链表(双向链表)的元素在内存中非连续存储,每个元素通过指针连接前后元素。链表在任意位置插入和删除元素的时间复杂度都是O(1),但随机访问需要遍历,时间复杂度为O(n)。
Go语言中并没有提供内置函数来删除切片,熟悉Java/PHP的都知道,在Java/PHP中都提供对list或数组的filter操作,那么在Go语言中就需要多用几行代码来实现删除切片中的指定元素了。
假设有个1~9的数组,要删除掉其中的偶数,那么这里先进行for循环,依次判断元素是否为偶数,然后将偶数的元素通过索引截取掉,下标i向左移动一位。
numbers := []int{1, 2, 3, 4, 5, 6, 7, 8, 9}
for i := 0; i < len(numbers); i++ {
if numbers[i]%2 == 0 { // 如果是偶数就删除
// 通过截取删除
numbers = append(numbers[:i], numbers[i+1:]...)
i--
}
}
fmt.Println(numbers) // [1 3 5 7 9]
切片,是一组可变长度的、同类元素的集合
与数组相比切片的长度是不固定的。切片可以追加元素,在追加元素时可能使切片的容量增大。
切片在使用前必须初始化,未初始化的切片的值为nil。
数组,是一组固定长度的、同类元素的集合。
固定长度,声明时指定长度,长度不可以改变。
同类元素,数组中的元素必须是同一类型的
专业企业官网建设,塑造企业形象,传递企业价值
系统软件开发,用心思考,用心设计,用心体验
打破技术瓶颈,让不堪重负的项目起死回生
构建全渠道一体化运营能力,实现全链路数字化
文案撰写、营销策划,专注品牌全案
一站式解决企业互联网营销痛点和难题
以技术的力量,改变互联网
联系我们