io.Seeker 通过 Seek(offset, whence) 灵活定位文件读写位置,offset 为偏移量,whence 指定参考点(SeekStart/Current/End),返回新绝对偏移量;不支持 seek 的对象(如管道)会报错,且需注意缓冲区一致性。

使用 io.Seeker 可以在文件中灵活跳转读写位置,核心是调用 Seek() 方法。它不负责读写,只改变“当前偏移量”,后续的 Read() 或 Write() 都从新位置开始。
理解 Seek 的三个参数:offset、whence
Seek(offset int64, whence int) (int64, error) 中:
- offset:偏移字节数(可正可负)
-
whence 决定 offset 的参考点,常用三个常量:
-
io.SeekStart:从文件开头算起(offset ≥ 0) -
io.SeekCurrent:从当前位置算起(offset 可正可负) -
io.SeekEnd:从文件末尾算起(offset 通常为负,如 -1 表示倒数第 1 字节)
-
常见定位操作示例
假设打开一个文件:f, _ := os.Open("data.txt")
-
跳到开头重新读:
f.Seek(0, io.SeekStart) -
跳过前 10 字节:
f.Seek(10, io.SeekStart) -
向后移动 5 字节:
f.Seek(5, io.SeekCurrent) -
回到上一个字节(比如重试解析):
f.Seek(-1, io.SeekCurrent) -
定位到倒数第 3 字节:
f.Seek(-3, io.SeekEnd)
注意 Seek 的返回值和边界行为
Seek() 返回的是**新的绝对偏移量**(从文件开头算),不是 offset 本身。它可能成功但返回值超出文件长度——这不报错,只是下次 Read() 会立刻返回 io.EOF。
立即学习“go语言免费学习笔记(深入)”;
- 若文件只有 100 字节,执行
f.Seek(200, io.SeekStart)会返回200,无错误 - 紧接着
buf := make([]byte, 1); f.Read(buf)会返回0, io.EOF - 想安全获取文件大小?用
f.Stat().Size(),别依赖 Seek 返回值判断长度
Seek 与 Read/Write 的配合要点
io.Seeker 常见于需要随机访问的场景(如解析二进制协议、日志回溯、断点续传)。注意:
- 对只读文件调用
Seek()没问题;但对管道、网络连接等非 seekable 对象会返回io.SeekError - 如果底层是
*os.File,Seek()是系统调用,开销低;但如果是包装过的 reader(如bufio.Reader),需先Reset()或用UnreadByte()等辅助,否则缓冲区内容和文件位置可能不一致 - 写入前建议先
Seek()定位,尤其用os.O_RDWR打开时,避免覆盖或追加意外
基本上就这些。掌握 offset 和 whence 的组合逻辑,再留意底层对象是否支持 seek,就能稳稳控制文件游标了。










