删除 for 中的切片元素
来源:Golang技术栈
时间:2023-04-13 15:21:12 137浏览 收藏
今天golang学习网给大家带来了《删除 for 中的切片元素》,其中涉及到的知识点包括golang等等,无论你是小白还是老手,都适合看一看哦~有好的建议也欢迎大家在评论留言,若是看完有所收获,也希望大家能多多点赞支持呀!一起加油学习~
问题内容
i
从 slice中删除元素并a
保留顺序的惯用方法似乎是:
a = append(a[:i], a[i+1:]...)
我想知道在循环中哪种方法是最好的方法。据我了解,不可能在以下范围内使用它:
for i := range a { // BAD if conditionMeets(a[i]) { a = append(a[:i], a[i+1:]...) } }
但是可以使用len(a)
. [编辑:这不起作用,请参阅下面的答案]
for i := 0; i有比使用
len
or更好或更惯用的方法append
吗?正确答案
您提出的解决方案不正确。问题是,当您从切片中删除一个元素时,所有后续元素都会被 移动 。但是循环不知道你改变了底层切片和循环变量(索引)像往常一样增加,即使在这种情况下它不应该因为你跳过一个元素。
如果切片包含两个相邻的元素,这两个元素都需要删除,则第二个元素将不会被检查并且不会被删除。
所以如果你删除一个元素,循环变量必须手动递减!让我们看一个例子:删除以 开头的单词
"a"
:func conditionMeets(s string) bool { return strings.HasPrefix(s, "a") }解决方案(在Go Playground上与以下所有其他示例一起尝试):
a := []string{"abc", "bbc", "aaa", "aoi", "ccc"} for i := 0; i输出:
[bbc ccc]或者更好: 使用向下循环,因此您不需要手动递减变量,因为在这种情况下,移位的元素位于切片的“已处理”部分。
a := []string{"abc", "bbc", "aaa", "aoi", "ccc"} for i := len(a) - 1; i >= 0; i-- { if conditionMeets(a[i]) { a = append(a[:i], a[i+1:]...) } } fmt.Println(a)输出是一样的。
替代多次移除
如果您必须删除“许多”元素,这可能会很慢,因为您必须进行大量复制(
append()
是否复制)。想象一下:你有一个包含 1000 个元素的切片;仅删除第一个元素需要将 999 个元素复制到前面。还将创建许多新的切片描述符:每次删除元素都会创建 2 个 新的切片描述符 (a[:i]
,a[i+1:]
) 并且a
必须更新 ( 的结果append()
)。在这种情况下,将不可移动元素复制到新切片可能更有效。一个有效的解决方案:
a := []string{"abc", "bbc", "aaa", "aoi", "ccc"} b := make([]string, len(a)) copied := 0 for _, s := range(a) { if !conditionMeets(s) { b[copied] = s copied++ } } b = b[:copied] fmt.Println(b)此解决方案分配与源相同长度的切片,因此不会执行新的分配(和复制)。此解决方案也可以使用
range
循环。如果您想要结果a
,请将结果分配给a
:a = b[:copied]
。输出是一样的。
就地替代许多移除(和一般用途)
我们还可以使用循环“就地”进行删除,方法是维护 2 个索引并在同一切片中分配(向前复制)不可删除的元素。
要记住的一件事是,我们应该将已删除元素的位置归零,以删除无法访问的值的引用,以便 GC 可以完成其工作。这也适用于其他解决方案,但仅在此处提及。
示例实现:
a := []string{"abc", "bbc", "aaa", "aoi", "ccc"} copied := 0 for i := 0; i输出是一样的。尝试Go Playground上的所有示例。
今天关于《删除 for 中的切片元素》的内容介绍就到此结束,如果有什么疑问或者建议,可以在golang学习网公众号下多多回复交流;文中若有不正之处,也希望回复留言以告知!
-
439 收藏
-
262 收藏
-
193 收藏
-
188 收藏
-
500 收藏
-
139 收藏
-
204 收藏
-
325 收藏
-
477 收藏
-
486 收藏
-
439 收藏
-
- 前端进阶之JavaScript设计模式
- 设计模式是开发人员在软件开发过程中面临一般问题时的解决方案,代表了最佳的实践。本课程的主打内容包括JS常见设计模式以及具体应用场景,打造一站式知识长龙服务,适合有JS基础的同学学习。
- 立即学习 542次学习
-
- GO语言核心编程课程
- 本课程采用真实案例,全面具体可落地,从理论到实践,一步一步将GO核心编程技术、编程思想、底层实现融会贯通,使学习者贴近时代脉搏,做IT互联网时代的弄潮儿。
- 立即学习 507次学习
-
- 简单聊聊mysql8与网络通信
- 如有问题加微信:Le-studyg;在课程中,我们将首先介绍MySQL8的新特性,包括性能优化、安全增强、新数据类型等,帮助学生快速熟悉MySQL8的最新功能。接着,我们将深入解析MySQL的网络通信机制,包括协议、连接管理、数据传输等,让
- 立即学习 497次学习
-
- JavaScript正则表达式基础与实战
- 在任何一门编程语言中,正则表达式,都是一项重要的知识,它提供了高效的字符串匹配与捕获机制,可以极大的简化程序设计。
- 立即学习 487次学习
-
- 从零制作响应式网站—Grid布局
- 本系列教程将展示从零制作一个假想的网络科技公司官网,分为导航,轮播,关于我们,成功案例,服务流程,团队介绍,数据部分,公司动态,底部信息等内容区块。网站整体采用CSSGrid布局,支持响应式,有流畅过渡和展现动画。
- 立即学习 484次学习