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() 的值提前缓存到局部变量中,再执行可能的删除操作,最后用缓存值更新循环变量。这种“先取后删”的模式确保了迭代的连续性。

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

知元AI
知元AI

AI智能语音聊天 对讲问答 AI绘画 AI写作 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结构体相关大全,想了解更多内容,请阅读专题下面的文章。

197

2025.06.09

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

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

190

2025.07.04

string转int
string转int

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

338

2023.08.02

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

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

542

2024.08.29

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

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

53

2025.08.29

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

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

197

2025.08.29

Java编译相关教程合集
Java编译相关教程合集

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

11

2026.01.21

C++多线程相关合集
C++多线程相关合集

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

4

2026.01.21

无人机驾驶证报考 uom民用无人机综合管理平台官网
无人机驾驶证报考 uom民用无人机综合管理平台官网

无人机驾驶证(CAAC执照)报考需年满16周岁,初中以上学历,身体健康(矫正视力1.0以上,无严重疾病),且无犯罪记录。个人需通过民航局授权的训练机构报名,经理论(法规、原理)、模拟飞行、实操(GPS/姿态模式)及地面站训练后考试合格,通常15-25天拿证。

16

2026.01.21

热门下载

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

精品课程

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

共32课时 | 4万人学习

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号