如何在 Go 中高效地从切片中删除元素?
php小编苹果为您介绍如何在 Go 中高效地从切片中删除元素。在 Go 语言中,删除切片中的元素是一个常见的操作,但是由于切片的特性,直接删除一个元素可能会导致切片长度的改变,从而影响后续的操作。为了高效地删除切片中的元素,我们可以利用切片的特性和一些内置函数来实现。下面将为您详细介绍几种常用的方法。
问题内容
有多种方法可以删除切片元素。但是,如果我有一个需要大量处理切片的应用程序怎么办? Go 切片对于添加新元素进行了很好的优化,但是有没有一种有效的方法可以从切片中删除元素(不仅是速度,而且还优化了内存)。
我知道 Go 1.21 中引入的 slices.Delete 函数,但在幕后它使用了以下众所周知的技术:
return append(s[:i], s[j:]...)
看起来在这种情况下底层数组不会减少。这对速度很有好处,但如果我们有很多元素(例如 100k 或 1M),然后将它们减少到很少(例如只有 10 个),该怎么办?看起来没有像用于增加切片容量的内存优化那样的内存优化。
当我们不需要保留切片中元素的顺序时,可以使用以下方法(转到游乐场链接):
func sliceDel[S ~[]E, E any](s S, i, j int) S { lastIdx := len(s) - (j - i) copy(s[i:], s[lastIdx:]) return s[:lastIdx] }
当我们有大切片和少量要删除的元素时,这会很有用(其背后的想法是复制少量切片元素)。
关于内存,两种情况下容量都是相同的并且不会减少。例如:
// Reduce slice almost to zero for i := 0; i < sliceSize/2-1; i++ { sl = sliceDel(sl, 0, 2) } fmt.Printf("len = %d, cap = %d", len(sl), cap(sl)) // Output: len = 2, cap = 100000 // Reduce slice almost to zero for i := 0; i < sliceSize/2-1; i++ { sl = slices.Delete(sl, 0, 2) } fmt.Printf("len = %d, cap = %d", len(sl), cap(sl)) // Output: len = 2, cap = 100000
那么,有没有办法优化内存使用呢?例如,如果切片的长度小于其容量的一半,则将容量减少一半。
我也想知道如何有效地做到这一点,例如这样的技术 s[:len(s):len(s)]
(完整切片表达式由 slices.Clip 使用)不会减少底层数组 - 它仅在切片结构中保存新容量,以避免在将新元素附加到子切片时重写父切片元素(正如本提案中提到的)。
解决方法
不存在“一般最佳”解决方案。您在问题中展示了多种方法,对于特定场景,每种方法都可能比其他方法更好。
如果您遇到这样的情况,当您想保留许多元素中的少数元素时,甚至不要开始删除这些元素。用这几个元素构建一个新切片。除了速度更快之外,这肯定也解决了内存问题。
除了分配和使用新切片之外,您无法通过使用完整切片表达式来减少内存使用量。只要存在对后备数组的引用,它就不会缩小(至少在当前的 Go 版本中不会)。如果您遇到分配了大后备数组但只使用其中一小部分的情况,则可以分配一个新切片并手动复制元素,以让大数组被垃圾收集。
还要考虑到,如果您有一个很大的切片,您可能需要从中删除许多元素,那么切片可能不是最好的数据结构。例如,您可以尝试使用链表,或者甚至可以尝试映射:从链表或映射中删除元素会快得多,映射还将提供快速 (O(n)
) 查找时间,如下所示好吧。
以上是如何在 Go 中高效地从切片中删除元素?的详细内容。更多信息请关注PHP中文网其他相关文章!

热AI工具

Undresser.AI Undress
人工智能驱动的应用程序,用于创建逼真的裸体照片

AI Clothes Remover
用于从照片中去除衣服的在线人工智能工具。

Undress AI Tool
免费脱衣服图片

Clothoff.io
AI脱衣机

AI Hentai Generator
免费生成ai无尽的。

热门文章

热工具

记事本++7.3.1
好用且免费的代码编辑器

SublimeText3汉化版
中文版,非常好用

禅工作室 13.0.1
功能强大的PHP集成开发环境

Dreamweaver CS6
视觉化网页开发工具

SublimeText3 Mac版
神级代码编辑软件(SublimeText3)

热门话题

OpenSSL,作为广泛应用于安全通信的开源库,提供了加密算法、密钥和证书管理等功能。然而,其历史版本中存在一些已知安全漏洞,其中一些危害极大。本文将重点介绍Debian系统中OpenSSL的常见漏洞及应对措施。DebianOpenSSL已知漏洞:OpenSSL曾出现过多个严重漏洞,例如:心脏出血漏洞(CVE-2014-0160):该漏洞影响OpenSSL1.0.1至1.0.1f以及1.0.2至1.0.2beta版本。攻击者可利用此漏洞未经授权读取服务器上的敏感信息,包括加密密钥等。

Go语言中用于浮点数运算的库介绍在Go语言(也称为Golang)中,进行浮点数的加减乘除运算时,如何确保精度是�...

Go爬虫Colly中的Queue线程问题探讨在使用Go语言的Colly爬虫库时,开发者常常会遇到关于线程和请求队列的问题。�...

本文讨论了GO编程中的GO FMT命令,该命令将代码格式化以遵守官方样式准则。它突出了GO FMT在维持代码一致性,可读性和降低样式辩论方面的重要性。 FO的最佳实践

本文介绍在Debian系统下监控PostgreSQL数据库的多种方法和工具,助您全面掌握数据库性能监控。一、利用PostgreSQL内置监控视图PostgreSQL自身提供多个视图用于监控数据库活动:pg_stat_activity:实时展现数据库活动,包括连接、查询和事务等信息。pg_stat_replication:监控复制状态,尤其适用于流复制集群。pg_stat_database:提供数据库统计信息,例如数据库大小、事务提交/回滚次数等关键指标。二、借助日志分析工具pgBadg

后端学习路径:从前端转型到后端的探索之旅作为一名从前端开发转型的后端初学者,你已经有了nodejs的基础,...
