0

0

如何在 Go 中安全地边遍历边删除链表元素

霞舞

霞舞

发布时间:2026-01-03 18:41:00

|

345人浏览过

|

来源于php中文网

原创

如何在 Go 中安全地边遍历边删除链表元素

go 的 `container/list` 中直接边遍历边删除会导致迭代中断,因为被删节点的 `next()` 返回 `nil`;正确做法是**提前保存下一个节点指针**,再执行删除操作。

Go 标准库的 container/list 是一个双向链表实现,其 Element 结构体不持有对链表的强引用,因此在遍历时若调用 l.Remove(e),该节点即从链表中解绑——此时 e.Next() 不再指向有效后继节点(可能为 nil 或已失效),导致 for e := l.Front(); e != nil; e = e.Next() 循环提前终止。

解决的核心思路是:将 e.Next() 的值提前缓存到局部变量中,再执行可能的删除操作,最后用缓存值更新循环变量。这种“先取后删”的模式确保了迭代的连续性。

以下是修正后的去重函数完整示例:

ChatGPT Writer
ChatGPT Writer

免费 Chrome 扩展程序,使用 ChatGPT AI 生成电子邮件和消息。

下载
func removeDuplicate(l *list.List) *list.List {
    seen := make(map[int]bool) // 建议使用局部变量而非全局 sMap,避免并发/复用问题
    var next *list.Element
    for e := l.Front(); e != nil; e = next {
        next = e.Next() // ✅ 关键:在任何可能修改 e 状态的操作前,先保存下一节点
        if val, ok := e.Value.(int); ok {
            fmt.Printf("VALUE: %d\n", val)
            if seen[val] {
                fmt.Printf("Deleting %d\n", val)
                l.Remove(e)
            } else {
                fmt.Printf("Adding new entry %d\n", val)
                seen[val] = true
            }
        }
    }
    return l
}

⚠️ 注意事项:

  • 永远不要在循环条件中依赖被删节点的方法调用(如 e.Next());
  • 使用 var next *list.Element 显式声明并初始化为 nil,避免未定义行为;
  • seen 映射建议定义在函数内,避免全局状态引发的竞态或副作用;
  • 类型断言 e.Value.(int) 应配合 ok 判断增强健壮性(生产环境推荐);
  • 若需保留首次出现的元素(如本例),逻辑天然成立;若需保留最后一次,则应反向遍历(Back() → Prev())并调整判断逻辑。

该模式不仅适用于去重,也适用于任意基于条件的过滤、清理或转换场景,是操作 container/list 时必须掌握的安全惯用法。

相关专题

更多
golang结构体相关大全
golang结构体相关大全

本专题整合了golang结构体相关大全,想了解更多内容,请阅读专题下面的文章。

194

2025.06.09

golang结构体方法
golang结构体方法

本专题整合了golang结构体相关内容,请阅读专题下面的文章了解更多。

186

2025.07.04

string转int
string转int

在编程中,我们经常会遇到需要将字符串(str)转换为整数(int)的情况。这可能是因为我们需要对字符串进行数值计算,或者需要将用户输入的字符串转换为整数进行处理。php中文网给大家带来了相关的教程以及文章,欢迎大家前来学习阅读。

314

2023.08.02

int占多少字节
int占多少字节

int占4个字节,意味着一个int变量可以存储范围在-2,147,483,648到2,147,483,647之间的整数值,在某些情况下也可能是2个字节或8个字节,int是一种常用的数据类型,用于表示整数,需要根据具体情况选择合适的数据类型,以确保程序的正确性和性能。本专题为大家提供相关的文章、下载、课程内容,供大家免费下载体验。

524

2024.08.29

c++怎么把double转成int
c++怎么把double转成int

本专题整合了 c++ double相关教程,阅读专题下面的文章了解更多详细内容。

49

2025.08.29

C++中int的含义
C++中int的含义

本专题整合了C++中int相关内容,阅读专题下面的文章了解更多详细内容。

191

2025.08.29

php代码编辑器入口汇总
php代码编辑器入口汇总

本文整理了主流PHP代码编辑器的官网入口及在线使用链接,阅读专题下面的文章了解更多详细内容。

26

2026.01.04

php代码编辑器地址汇总
php代码编辑器地址汇总

本文整理了主流PHP代码编辑器的官网入口及在线使用链接,阅读专题下面的文章了解更多详细内容。

0

2026.01.04

Excel制作交互图表的方法 Excel交互式的图表教程大全
Excel制作交互图表的方法 Excel交互式的图表教程大全

Excel交互式图表可通过四种方法实现:一、用切片器控制数据透视图;二、结合下拉列表与INDEX-MATCH动态引用;三、用选项按钮绑定图表系列;四、利用动态命名区域配合OFFSET函数。本专题为大家提供相关的文章、下载、课程内容,供大家免费下载体验。

181

2026.01.04

热门下载

更多
网站特效
/
网站源码
/
网站素材
/
前端模板

精品课程

更多
相关推荐
/
热门推荐
/
最新课程
Go 教程
Go 教程

共32课时 | 3.3万人学习

Go语言实战之 GraphQL
Go语言实战之 GraphQL

共10课时 | 0.8万人学习

关于我们 免责申明 举报中心 意见反馈 讲师合作 广告合作 最新更新
php中文网:公益在线php培训,帮助PHP学习者快速成长!
关注服务号 技术交流群
PHP中文网订阅号
每天精选资源文章推送

Copyright 2014-2026 https://www.php.cn/ All Rights Reserved | php.cn | 湘ICP备2023035733号