并发写入文件
来源:Golang技术栈
时间:2023-03-22 19:58:46 276浏览 收藏
今天golang学习网给大家带来了《并发写入文件》,其中涉及到的知识点包括golang等等,无论你是小白还是老手,都适合看一看哦~有好的建议也欢迎大家在评论留言,若是看完有所收获,也希望大家能多多点赞支持呀!一起加油学习~
问题内容
在 go 中,如何控制对文本文件的并发写入?
我问这个是因为我将有多个 goroutine 使用同一个文件处理程序写入一个文本文件。
我写了这段代码来尝试看看会发生什么,但我不确定我是否“正确”地做到了:
package main import ( "os" "sync" "fmt" "time" "math/rand" "math" ) func WriteToFile( i int, f *os.File, w *sync.WaitGroup ){ //sleep for either 200 or 201 milliseconds randSleep := int( math.Floor( 200 + ( 2 * rand.Float64() ) ) ) fmt.Printf( "Thread %d waiting %d\n", i, randSleep ) time.Sleep( time.Duration(randSleep) * time.Millisecond ) //write to the file fmt.Fprintf( f, "Printing out: %d\n", i ) //write to stdout fmt.Printf( "Printing out: %d\n", i ) w.Done() } func main() { rand.Seed( time.Now().UnixNano() ) d, err := os.Getwd() if err != nil { fmt.Println( err ) } filename := d + "/log.txt" f, err := os.OpenFile( filename, os.O_CREATE | os.O_WRONLY | os.O_TRUNC, 0666 ) if err != nil { fmt.Println( err ) } var w *sync.WaitGroup = new(sync.WaitGroup) w.Add( 10 ) //start 10 writers to the file for i:=1; i我有一半期望输出会在文件中显示类似这样的内容,而不是我得到的连贯输出:
Printing Printing out: 2 out: 5 Poriuntitng: 6本质上,我预计角色会因为缺乏同步而语无伦次地交织在一起。我没有编写可以诱使这种行为的代码吗?或者是在调用
fmt.Fprintf
同步写作期间的某种机制?正确答案
有很多方法可以控制并发访问。最简单的是使用Mutex:
var mu sync.Mutex func WriteToFile( i int, f *os.File, w *sync.WaitGroup ){ mu.Lock() defer mu.Unlock() // etc... }至于为什么您没有看到问题,Go 使用操作系统调用来实现文件访问,并且这些系统调用是[线程安全](http://man7.org/linux/man- pages/man2/write.2.html)的(强调添加):
根据 POSIX.1-2008/SUSv4 Section XSI 2.9.7(“线程与常规文件操作的交互”):
以下所有函数在对常规文件或符号链接进行操作时,在 POSIX.1-2008 中指定的效果中相互之间应是原子的: ...
随后列出的 API 包括 write() 和 writev(2)。跨线程(和进程)应该是原子的影响之一是文件偏移量的更新。但是,在 3.14 之前的 Linux 上,情况并非如此:如果共享打开文件描述(参见 open(2))的两个进程同时执行 write()(或 writev(2)),则 I /O 操作在更新文件偏移方面不是原子的,因此两个进程输出的数据块可能(不正确地)重叠。 此问题已在 Linux 3.14 中修复。
不过我还是会使用锁,因为 Go 代码不是自动线程安全的。(两个 goroutine 修改同一个变量会导致奇怪的行为)
理论要掌握,实操不能落!以上关于《并发写入文件》的详细介绍,大家都掌握了吧!如果想要继续提升自己的能力,那么就来关注golang学习网公众号吧!
-
439 收藏
-
262 收藏
-
193 收藏
-
188 收藏
-
500 收藏
-
139 收藏
-
204 收藏
-
325 收藏
-
477 收藏
-
486 收藏
-
439 收藏
-
- 前端进阶之JavaScript设计模式
- 设计模式是开发人员在软件开发过程中面临一般问题时的解决方案,代表了最佳的实践。本课程的主打内容包括JS常见设计模式以及具体应用场景,打造一站式知识长龙服务,适合有JS基础的同学学习。
- 立即学习 542次学习
-
- GO语言核心编程课程
- 本课程采用真实案例,全面具体可落地,从理论到实践,一步一步将GO核心编程技术、编程思想、底层实现融会贯通,使学习者贴近时代脉搏,做IT互联网时代的弄潮儿。
- 立即学习 507次学习
-
- 简单聊聊mysql8与网络通信
- 如有问题加微信:Le-studyg;在课程中,我们将首先介绍MySQL8的新特性,包括性能优化、安全增强、新数据类型等,帮助学生快速熟悉MySQL8的最新功能。接着,我们将深入解析MySQL的网络通信机制,包括协议、连接管理、数据传输等,让
- 立即学习 497次学习
-
- JavaScript正则表达式基础与实战
- 在任何一门编程语言中,正则表达式,都是一项重要的知识,它提供了高效的字符串匹配与捕获机制,可以极大的简化程序设计。
- 立即学习 487次学习
-
- 从零制作响应式网站—Grid布局
- 本系列教程将展示从零制作一个假想的网络科技公司官网,分为导航,轮播,关于我们,成功案例,服务流程,团队介绍,数据部分,公司动态,底部信息等内容区块。网站整体采用CSSGrid布局,支持响应式,有流畅过渡和展现动画。
- 立即学习 484次学习