惯性聚合 高效追踪和阅读你感兴趣的博客、新闻、科技资讯
阅读原文 在惯性聚合中打开

推荐订阅源

V
Vulnerabilities – Threatpost
L
LINUX DO - 热门话题
F
Fox-IT International blog
C
Cisco Blogs
C
CERT Recently Published Vulnerability Notes
T
Tor Project blog
Malwarebytes
Malwarebytes
Latest news
Latest news
D
Darknet – Hacking Tools, Hacker News & Cyber Security
SecWiki News
SecWiki News
N
News and Events Feed by Topic
T
True Tiger Recordings
www.infosecurity-magazine.com
www.infosecurity-magazine.com
美团技术团队
P
Palo Alto Networks Blog
V
V2EX - 技术
AWS News Blog
AWS News Blog
A
About on SuperTechFans
Microsoft Azure Blog
Microsoft Azure Blog
量子位
博客园 - 【当耐特】
P
Proofpoint News Feed
N
News and Events Feed by Topic
博客园 - 司徒正美
U
Unit 42
G
Google Developers Blog
阮一峰的网络日志
阮一峰的网络日志
Schneier on Security
Schneier on Security
G
GRAHAM CLULEY
O
OpenAI News
T
The Blog of Author Tim Ferriss
F
Future of Privacy Forum
IntelliJ IDEA : IntelliJ IDEA – the Leading IDE for Professional Development in Java and Kotlin | The JetBrains Blog
IntelliJ IDEA : IntelliJ IDEA – the Leading IDE for Professional Development in Java and Kotlin | The JetBrains Blog
Blog — PlanetScale
Blog — PlanetScale
人人都是产品经理
人人都是产品经理
奇客Solidot–传递最新科技情报
奇客Solidot–传递最新科技情报
N
News | PayPal Newsroom
V
Visual Studio Blog
V
V2EX
Simon Willison's Weblog
Simon Willison's Weblog
Microsoft Security Blog
Microsoft Security Blog
C
Cyber Attacks, Cyber Crime and Cyber Security
T
Threat Research - Cisco Blogs
Spread Privacy
Spread Privacy
N
Netflix TechBlog - Medium
Threat Intelligence Blog | Flashpoint
Threat Intelligence Blog | Flashpoint
D
Docker
小众软件
小众软件
H
Hackread – Cybersecurity News, Data Breaches, AI and More
I
Intezer

郑文峰的博客

使用dify对接飞书多维表格 使用n8n对接飞书多维表格 服务启动时出现 OOM Bug 通缉令 一次服务升级时pg表DDL执行超时失败 Go语言高性能编程 Go语言高效IO缓冲技术详解 Go语言延迟初始化(Lazy Initialization)最佳实践 Go语言字符串拼接性能对比与优化指南 Go语言结构体内存对齐完全指南 Go语言空结构体:零内存消耗的高效编程 Go语言堆栈分配与逃逸分析深度解析 Go语言原子操作完全指南 Go语言内存预分配完全指南 Go语言不可变数据共享:无锁并发编程实践 Go语言零拷贝技术完全指南 Go语言Interface Boxing原理与性能优化指南 Go协程池深度解析:原理、实现与最佳实践 使用etcd分布式锁导致的协程泄露与死锁问题 基于pre-commit的Python代码规范落地实践 初识 MCP Server pulsar阻塞导致logstash无法接入日志 django-prometheus使用及源码分析 kube-proxy源码分析 kubernetes service如何通过iptables转发 tcp缓存引起的日志丢失 django-apschedule定时任务异常停止 理解calico容器网络通信方案原理 理解flannel的三种容器网络方案原理 理解Linux IPIP隧道 理解VXLAN网络 理解Linux TunTap设备 快速了解iptables kafka中listener和advertised.listeners的作用 django rest_framework 分页 django后端服务、logstash和flink接入VictoriaMetrics指标监控 python中import原理 docker容器单机网络 手动实现docker容器bridge网络模型 mysql之MVCC原理 mysql之日志 使用java开发logstash的filter插件 使用python实现单例模式的三种方式 redis之缓存 redis之分片集群 redis之哨兵机制 redis之主从库同步 redis之持久化 redis之五种基本数据类型 go中如何处理error pod中将代码与运行环境分离 友链 ddt源码分析 python装饰器的使用方法 读书笔记:如何阅读一本书 使用ddt实现unittest的参数化测试 分布式锁 使用kubeadm安装k8s 优化gin表单的错误提示信息 gin中validator模块的源码分析 go简单使用grpc python简单使用grpc k8s之PV、PVC和StorageClass k8s之StatefulSet k8s之DaemonSet k8s之Job和CronJob k8s之ConfigMap和Secret k8s之Service k8s之Pod k8s之Deployment 容器的本质 docker容器 python迭代器与生成器 python元编程 python垃圾回收机制 python上下文管理器 django rest_framework使用jwt django rest_framework异常处理 django rest_framework 自定义文档 django压缩文件下载 django rest_framework使用pytest单元测试 django restframework choice 自定义输出数据 django Filtering 使用 django viewset 和 Router 配合使用时报的错 django model的序列化 django中使用AbStractUser django.core.exceptions.ImproperlyConfigured Application labels aren't unique, duplicates users django 中 media配置 django 外键引用自身和on_delete参数 django 警告 while time zone support is active Flask使用flask_socketio实现websocket flask结合mongo tornado 文件上传 tornado 使用jwt完成用户异步认证 tornado 用户密码 bcrypt加密 tornado 结合wtforms使用表单操作 tornado finish和write区别 tornado 使用peewee-async 完成异步orm数据库操作 pyspark streaming简介 和 消费 kafka示例 使用hue创建ozzie的pyspark action workflow
Go语言遍历性能深度解析:从原理到优化实践
2025-06-14 · via 郑文峰的博客

# 前言

在Go语言中,遍历是日常开发中最常见的操作之一。不同的遍历方式会对性能产生显著影响。本文将深入分析:

  1. 基本切片遍历的性能特点
  2. 结构体切片的遍历优化
  3. 指针切片的性能优势
  4. 实际场景中的最佳实践

# 三种遍历方式对比

Go语言中主要有三种遍历切片的方式:

// 1. 索引遍历
for i := 0; i < len(slice); i++ {
    // 使用slice[i]
}

// 2. range遍历值
for _, v := range slice {
    // 使用v
}

// 3. range遍历索引和值
for i, v := range slice {
    // 使用i和v
}

1
2
3
4
5
6
7
8
9
10
11
12
13
14

# []int Benchmark测试

func BenchmarkIndexLoop(b *testing.B) {
    slice := make([]int, 1000)
    for i := 0; i < b.N; i++ {
        for j := 0; j < len(slice); j++ {
            _ = slice[j]
        }
    }
}

func BenchmarkRangeValue(b *testing.B) {
    slice := make([]int, 1000)
    for i := 0; i < b.N; i++ {
        for _, v := range slice {
            _ = v
        }
    }
}

func BenchmarkRangeIndexValue(b *testing.B) {
    slice := make([]int, 1000)
    for i := 0; i < b.N; i++ {
        for j, v := range slice {
            _, _ = j, v
        }
    }
}

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26

测试结果如下,可以发现三者性能接近,基本相差不大。

% go test -bench=. -benchmem .
goos: darwin
goarch: arm64
pkg: main/demo
cpu: Apple M4 Pro
BenchmarkIndexLoop-12            4721586               235.7 ns/op             0 B/op          0 allocs/op
BenchmarkRangeValue-12           5085130               234.3 ns/op             0 B/op          0 allocs/op
BenchmarkRangeIndexValue-12      5101604               233.9 ns/op             0 B/op          0 allocs/op
PASS
ok      main/demo       4.560s

1
2
3
4
5
6
7
8
9
10

# []struct Benchmark 测试


type Item struct {
	ID   int
	Data [4096]byte // 增加数据量以放大性能差异
}

func BenchmarkStructIndex(b *testing.B) {
	var slice [1000]Item
	for i := 0; i < b.N; i++ {
		var tmp int
		for j := 0; j < len(slice); j++ {
			tmp = slice[j].ID
		}
		_ = tmp
	}
}

func BenchmarkStructRangeValue(b *testing.B) {
	var slice [1000]Item
	for i := 0; i < b.N; i++ {
		var tmp int
		for _, v := range slice {
			tmp = v.ID
		}
		_ = tmp
	}
}

func BenchmarkStructRangeIndexValue(b *testing.B) {
	var slice [1000]Item
	for i := 0; i < b.N; i++ {
		var tmp int
		for j := range slice {
			tmp = slice[j].ID
		}
		_ = tmp
	}
}

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38

运行结果如下,可以发现通过索引的方式取值的两种方式相差不大,但是直接取值的方式性能差了 500 多倍,这是因为直接取值时会进行数据的复制,而索引取值不会进行数据的复制。

$ go test  -bench=. -benchmem . 
goos: darwin
goarch: arm64
pkg: main/demo
cpu: Apple M4 Pro
BenchmarkStructIndex-12                  4735326               237.9 ns/op             0 B/op          0 allocs/op
BenchmarkStructRangeValue-12               17096             69135 ns/op               0 B/op          0 allocs/op
BenchmarkStructRangeIndexValue-12        5123646               234.6 ns/op             0 B/op          0 allocs/op
PASS
ok      main/demo       4.983s

1
2
3
4
5
6
7
8
9
10

# []*struch Benchmark 测试


type Item struct {
	ID   int
	Data [4096]byte // 增加数据量以放大性能差异
}

func BenchmarkStructIndex(b *testing.B) {
	var slice [1000]*Item
	for i := range slice {
		slice[i] = &Item{}
	}
	for i := 0; i < b.N; i++ {
		var tmp int
		for j := 0; j < len(slice); j++ {
			tmp = slice[j].ID
		}
		_ = tmp
	}
}

func BenchmarkStructRangeValue(b *testing.B) {
	var slice [1000]*Item
	for i := range slice {
		slice[i] = &Item{}
	}
	for i := 0; i < b.N; i++ {
		var tmp int
		for _, v := range slice {
			tmp = v.ID
		}
		_ = tmp
	}
}

func BenchmarkStructRangeIndexValue(b *testing.B) {
	var slice [1000]*Item
	for i := range slice {
		slice[i] = &Item{}
	}
	for i := 0; i < b.N; i++ {
		var tmp int
		for j := range slice {
			tmp = slice[j].ID
		}
		_ = tmp
	}
}

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47

运行结果如下,三者性能相近,但是有个好处是可以直接修改指针对应结构体的值。

go test  -bench=. -benchmem .
goos: darwin
goarch: arm64
pkg: main/demo
cpu: Apple M4 Pro
BenchmarkStructIndex-12                  1864108               656.9 ns/op             2 B/op          0 allocs/op
BenchmarkStructRangeValue-12             1577792               748.6 ns/op             3 B/op          0 allocs/op
BenchmarkStructRangeIndexValue-12        1843874               661.4 ns/op             2 B/op          0 allocs/op
PASS
ok      main/demo       5.995s

1
2
3
4
5
6
7
8
9
10

# 总结

根据测试结果,我们得出以下结论:

  1. 基础类型切片:三种遍历方式性能相当,可按编码习惯选择
  2. 大结构体切片
    • 避免使用for _, v := range直接值遍历
    • 优先使用索引遍历或range索引遍历
  3. 需要修改元素时
    • 使用指针切片([]*T)性能接近且更灵活
  4. 性能关键路径
    • 对大结构体集合操作,索引遍历性能最优
    • 对小结构体或基础类型,差异可忽略

记住:没有绝对的最佳方式,只有最适合当前场景的选择!