前言

很多时候需要将遍历对象中去掉某些元素,或者往遍历对象中添加元素,这时候就需要小心操作了。

对于go语言中的一些注意事项我做了总结和示例,留下点笔记。

遍历切片

  1. 遍历切片时去掉元素,错误示例:
func main() {
arr := []int{1, 2, 3, 4}
for i := range arr {
if arr[i] == 3 {
arr = append(arr[:i], arr[i+1:]...)
}
}
fmt.Println(arr)
}

最终报错panic: runtime error: index out of range [3] with length 3,因为range在迭代时已经确定i的范围为[0,len(arr))的左闭右开的区间。

但是当满足arr[i] == 3时对arr进行了修改,缩短了arr的长度,此时len(arr)=3,最大下标为2,因此当执行arr[3]时会报错。

  1. 遍历切片时去掉元素,不会报错,但不建议的写法:
func main() {
arr := []int{1, 2, 3, 4}
for i, v := range arr {
fmt.Println(i, v)
if v == 3 {
arr = append(arr[:i], arr[i+1:]...)
// arr = []int{4, 5, 6, 7} // 可以将上一行代码替换,看最终遍历的i,v情况
}
}
fmt.Println(arr)
}

还是回到range的用法,当执行for循环时就已经确定(i,v)的遍历元素值,及时循环过程中修改arr,也不会改变for要遍历的(i,v)值。

可以将上面代码修改一下,看下在循环中改变arr值时,后面遍历的(i,v)是不会随着arr的改变而改变的。

  1. 遍历切片时去掉元素,建议写法:
func main() {
arr := []int{1, 2, 3, 4}
for i := 0; i < len(arr); i++ {
fmt.Println(i, arr[i])
if arr[i] == 3 {
arr = append(arr[:i], arr[i+1:]...)
i--
}
}
fmt.Println(arr)
}

该方案只修改i的值,在删除元素时进行i--,可以确保遍历arr没有问题,而且每次通过arr[i]获取切片值不存在问题。

当然用该方式也可以在遍历时添加元素,只要i也对应变化就没问题。

遍历map

  1. 遍历map时去掉元素,可参考官方示例,可看下官方描述,下面这种方案是安全的。
for key := range m {
if key.expired() {
delete(m, key)
}
}
  1. 清空map所有元素,如下第一种是省事的写法,第二种不会产生新的对象,用哪种看个人喜好吧。
m = make(map[int]int) // 可以产生一个新对象,旧对象等着被垃圾回收

for k := range m {
delete(m,k) // 循环遍历并删除map所有元素,好处是map缓存还在,下次添加时可直接使用缓存
}

总结

关于切片遍历时进行操作需要注意一些坑。

关于map遍历时进行操作相对坑少点,不过遍历map需要修改元素时,map的value要为指针类型,这点得谨记。

golang遍历时修改被遍历对象的更多相关文章

  1. 21、List遍历时修改元素的问题

    List迭代时修改元素的问题 请编写代码完成以下需求:判断一个List里面是否包含monkey,如果包含的话,向集合中添加1024这个字符串.‘ package com.monkey1024.list ...

  2. List集合遍历时修改元素出现并发修改异常总结

    什么是并发修改异常: 当我们在遍历实现了collection接口与iterator接口的集合时(List.Set.Map), 我们可以通过遍历索引也可以通过迭代器进行遍历.在我们使用迭代器进行遍历集合 ...

  3. 不要在遍历子结点时修改parent

    [不要在遍历子结点时修改parent] 在用for/foreach遍历子结点时,如果在这过程中有改变子结点的parent,会导致不可预料的结果.我所遇到的问题是,在此种情况下,并非所有的子结点都能遍历 ...

  4. C++进阶 STL(3) 第三天 函数对象适配器、常用遍历算法、常用排序算法、常用算数生成算法、常用集合算法、 distance_逆序遍历_修改容器元素

    01昨天课程回顾 02函数对象适配器 函数适配器是用来让一个函数对象表现出另外一种类型的函数对象的特征.因为,许多情况下,我们所持有的函数对象或普通函数的参数个数或是返回值类型并不是我们想要的,这时候 ...

  5. JAVA的Hashtable在遍历时的迭代器线程问题

    这篇博客主要讲什么 Hashtable及其内部类的部分源码分析 Hashtable在遍历时的java.util.ConcurrentModificationException异常的来由和解决 单机在内 ...

  6. 【原理探究】女朋友问我ArrayList遍历时删除元素的正确姿势是什么?

    简介 我们在项目开发过程中,经常会有需求需要删除ArrayList中的某个元素,而使用不正确的删除方式,就有可能抛出异常.或者在面试中,会遇到面试官询问遍历时如何正常删除元素.所以在本篇文章中,我们会 ...

  7. jackson中JSON字符串节点遍历和修改

    有些场景下,在实现一些基础服务和拦截器的时候,我们可能需要在不知道JSON字符串所属对象类型的情况下,对JSON字符串中的某些属性进行遍历和修改,比如,设置或查询一些报文头字段. 在jackson中, ...

  8. Java遍历时删除List、Set、Map中的元素(源码分析)

    在对List.Set.Map执行遍历删除或添加等改变集合个数的操作时,不能使用普通的while.for循环或增强for.会抛出ConcurrentModificationException异常或者没有 ...

  9. 【Java】List遍历时删除元素的正确方式

    当要删除ArrayList里面的某个元素,一不注意就容易出bug.今天就给大家说一下在ArrayList循环遍历并删除元素的问题.首先请看下面的例子: import java.util.ArrayLi ...

随机推荐

  1. 转:minhash

    Minhash算法及其应用 一.引言 MinHash算法属于Locality Sensitive Hashing,用于快速估计两个集合的相似度.最早由Broder Andrei Z. 在1997年提出 ...

  2. AcWing 326. XOR和路径

    大型补档计划 题目链接 如果整体来做,发现既有加法,也有整体异或,这样不容易搞. 考虑异或,各个位置互不干扰,按位考虑一下. 枚举每一位 \(k\) 发现如果设 \(f[u]\) 为这一位的期望结果还 ...

  3. sqli-labs less-7(文件读写)

    less-7 dump into outfile(文件读写) 通俗的来讲,就是通过outfile传入一句话木马到网站目录里,然后用菜刀或者蚁剑等连接 过程: 输入id=?判断闭合类型 页面上提示了使用 ...

  4. docker redis 设置和使用

    1 开启docker 拉取redis镜像 1.1 桌面版docker   在镜像所在位置命令行执行 docker load -i redis.tar 1.2 开启redis docker run -p ...

  5. 微信小程序 rich-text 修改照片

    <view> <rich-text nodes="{{delcon}}" /> </view> data: { delcon:'' }, var ...

  6. Grafana 备份恢复教程

    原文链接:https://fuckcloudnative.io/posts/how-to-back-up-all-of-your-grafana-dashboards/ 目前我们 k8s 集群的 Gr ...

  7. Spark性能调优篇一之任务提交参数调整

    问题一:有哪些资源可以分配给spark作业使用? 答案:executor个数,cpu per exector(每个executor可使用的CPU个数),memory per exector(每个exe ...

  8. SSRF深入学习

    爆出来的直接关于SSRF的漏洞有俩,①是weblogic,②是discuzz SSRF漏洞最主要的部分并不是SSRF 探测内网,而是可以写shell,反弹shell,虽然很多厂家把它归为低危漏洞,仔细 ...

  9. JDK8新特性详解(一)

    虽然JDK8已经出来了N久,其新特性也在日益改变着我们的编码习惯和风格.虽然有些新特性用起来很顺手,但是总是傻傻分不清到底是哪个版本的.趁今天有时间,我们就来总结一下,JDK8有哪些能提升我们开发效率 ...

  10. 图解 IP 基础知识!

    我把自己以往的文章汇总成为了 Github ,欢迎各位大佬 star https://github.com/crisxuan/bestJavaer IP 协议 路由器对分组进行转发后,就会把数据包传到 ...