0

0

Golang并发写文件要注意什么_Golang文件并发安全实战

P粉602998670

P粉602998670

发布时间:2026-01-31 10:32:18

|

272人浏览过

|

来源于php中文网

原创

多个goroutine并发调用os.WriteFile会覆盖彼此内容,因其原子性仅限单次调用;写同一文件需加锁或改用其他方案;os.File.Write非线程安全,须显式同步;bufio.Writer不支持并发共享,应按goroutine隔离或用sync.Pool复用。

golang并发写文件要注意什么_golang文件并发安全实战

多个 goroutine 直接 os.WriteFile 会覆盖彼此内容

Go 标准库os.WriteFile 是原子写入(先写临时文件再 rename),但「原子」只保证单次调用不被中断,不保证多 goroutine 并发调用时的顺序或互斥。如果 10 个 goroutine 同时对同一路径调用 os.WriteFile,最终文件内容大概率是最后一次写入的结果——前面 9 次全被覆盖。

  • 这不是 bug,是设计使然:它没义务帮你同步
  • 适用场景仅限「每个 goroutine 写不同文件」,比如日志按 ID 分片:log_user_123.jsonlog_user_456.json
  • 若必须写同一文件(如汇总统计),得自己加锁或换方案

os.OpenFile + file.Write 时必须显式加锁

打开一个共享 *os.File 后,并发调用其 Write 方法会导致数据错乱甚至 panic(底层 write(2) 系统调用不是线程安全的)。Go 的 os.File 本身不带内部锁,Write 只是转发系统调用。

  • 最简方式:用 sync.Mutex 包裹 file.Write 调用
  • 注意别锁住 file.Close —— 关闭后其他 goroutine 再 Write 会 panic:write: bad file descriptor
  • 避免在循环里反复 OpenFile(..., os.O_WRONLY|os.O_APPEND):开销大,且 O_APPEND 在多数文件系统上能保证追加原子性,但仍有极小概率因内核缓冲导致行交错(尤其写入非完整行时)

bufio.Writersync.Pool 配合能缓解频繁 I/O 压力

直接对 *os.File 频繁 Write 小数据(比如每条日志一行),系统调用开销高。用 bufio.Writer 缓冲后再刷盘更高效,但要注意:它不是并发安全的,不能多个 goroutine 共享同一个实例。

牛NIUCMS本地O2O系统
牛NIUCMS本地O2O系统

牛NIUCMS本地O2O系统是一个以php+mysql进行开发的o2o网站系统。NIUCMS是一款强大的网站管理系统。支持智慧城市、智慧小区、智慧乡村、本地生活门户、本地O2O平台的构建。请注意以下几点:1、这套源码必须要服务器支持伪静态,是支持.htaccess规则的伪静态,一般Apache服务器支持,别搞的下载回去以后说什么缺 少文件,其实源码并非缺少文件。2、这套源码请在php 5.4环境下

下载
  • 推荐做法:每个 goroutine 自己配一个 bufio.Writer,写完调 Flush
  • 若创建太频繁,可用 sync.Pool 复用 bufio.Writer 实例(注意:Pool.Get 返回的 writer 缓冲区可能非空,务必重置:wr.Reset(file)
  • 别省略 Flush:不调用就可能丢数据;也别在 defer 里无条件 Flush——万一 writer 还没写任何内容,Flush 会触发一次空系统调用

写入失败时,io.EOF 不代表磁盘满,syscall.ENOSPC 才是

并发写文件出错时,错误类型比错误字符串更可靠。常见误区是检查 err == io.EOF,但它只在读取到流末尾时出现,写入过程根本不会返回这个值。

立即学习go语言免费学习笔记(深入)”;

  • 磁盘满的真实信号是 syscall.Errno == syscall.ENOSPC(需导入 golang.org/x/sys/unix 或用 errors.Is(err, syscall.ENOSPC)
  • 权限不足对应 syscall.EACCES,路径不存在是 syscall.ENOENT
  • 并发下还可能出现 syscall.EAGAINsyscall.EWOULDBLOCK(尤其写 pipe 或 socket),但普通文件少见
实际最难处理的不是并发逻辑本身,而是「写入成功但没落盘」——比如用 O_SYNC 开标志又忘了处理对应性能暴跌,或者依赖 fsync 却没检查它的返回值。这些细节一旦漏掉,在高负载下才会暴露。

热门AI工具

更多
DeepSeek
DeepSeek

幻方量化公司旗下的开源大模型平台

豆包大模型
豆包大模型

字节跳动自主研发的一系列大型语言模型

通义千问
通义千问

阿里巴巴推出的全能AI助手

腾讯元宝
腾讯元宝

腾讯混元平台推出的AI助手

文心一言
文心一言

文心一言是百度开发的AI聊天机器人,通过对话可以生成各种形式的内容。

讯飞写作
讯飞写作

基于讯飞星火大模型的AI写作工具,可以快速生成新闻稿件、品宣文案、工作总结、心得体会等各种文文稿

即梦AI
即梦AI

一站式AI创作平台,免费AI图片和视频生成。

ChatGPT
ChatGPT

最最强大的AI聊天机器人程序,ChatGPT不单是聊天机器人,还能进行撰写邮件、视频脚本、文案、翻译、代码等任务。

相关专题

更多
golang如何定义变量
golang如何定义变量

golang定义变量的方法:1、声明变量并赋予初始值“var age int =值”;2、声明变量但不赋初始值“var age int”;3、使用短变量声明“age :=值”等等。本专题为大家提供相关的文章、下载、课程内容,供大家免费下载体验。

182

2024.02.23

golang有哪些数据转换方法
golang有哪些数据转换方法

golang数据转换方法:1、类型转换操作符;2、类型断言;3、字符串和数字之间的转换;4、JSON序列化和反序列化;5、使用标准库进行数据转换;6、使用第三方库进行数据转换;7、自定义数据转换函数。本专题为大家提供相关的文章、下载、课程内容,供大家免费下载体验。

229

2024.02.23

golang常用库有哪些
golang常用库有哪些

golang常用库有:1、标准库;2、字符串处理库;3、网络库;4、加密库;5、压缩库;6、xml和json解析库;7、日期和时间库;8、数据库操作库;9、文件操作库;10、图像处理库。本专题为大家提供相关的文章、下载、课程内容,供大家免费下载体验。

343

2024.02.23

golang和python的区别是什么
golang和python的区别是什么

golang和python的区别是:1、golang是一种编译型语言,而python是一种解释型语言;2、golang天生支持并发编程,而python对并发与并行的支持相对较弱等等。本专题为大家提供相关的文章、下载、课程内容,供大家免费下载体验。

210

2024.03.05

golang是免费的吗
golang是免费的吗

golang是免费的。golang是google开发的一种静态强类型、编译型、并发型,并具有垃圾回收功能的开源编程语言,采用bsd开源协议。本专题为大家提供相关的文章、下载、课程内容,供大家免费下载体验。

396

2024.05.21

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

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

240

2025.06.09

golang相关判断方法
golang相关判断方法

本专题整合了golang相关判断方法,想了解更详细的相关内容,请阅读下面的文章。

194

2025.06.10

golang数组使用方法
golang数组使用方法

本专题整合了golang数组用法,想了解更多的相关内容,请阅读专题下面的文章。

478

2025.06.17

漫画防走失登陆入口大全
漫画防走失登陆入口大全

2026最新漫画防走失登录入口合集,汇总多个稳定可用网址,助你畅享高清无广告漫画阅读体验。阅读专题下面的文章了解更多详细内容。

0

2026.01.31

热门下载

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

精品课程

更多
相关推荐
/
热门推荐
/
最新课程
WEB前端教程【HTML5+CSS3+JS】
WEB前端教程【HTML5+CSS3+JS】

共101课时 | 8.6万人学习

JS进阶与BootStrap学习
JS进阶与BootStrap学习

共39课时 | 3.2万人学习

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

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