登录
首页 >  Golang >  Go问答

为何这段代码没有满足并发要求?

来源:stackoverflow

时间:2024-03-01 12:00:30 112浏览 收藏

亲爱的编程学习爱好者,如果你点开了这篇文章,说明你对《为何这段代码没有满足并发要求?》很感兴趣。本篇文章就来给大家详细解析一下,主要介绍一下,希望所有认真读完的童鞋们,都有实质性的提高。

问题内容

我有这个 go 代码,它遍历目录文件树并生成其中每个文件的 md5 哈希值,并将结果写入输出文件中。

package main

import (
    "crypto/md5"
    "encoding/hex"
    "fmt"
    "io"
    "os"
    "path/filepath"
    "sync"
)

func main() {

    filePath := os.Args[1]
    output := os.Args[2]
    wg := &sync.WaitGroup{}

    err := filepath.Walk(filePath, func(path string, info os.FileInfo, err error) error {
        if !info.IsDir() {
            wg.Add(1)
            go func(path string) {
                md5Sum, _ := md5File(path)
                if err := writeToFile(path, md5Sum, output); err != nil {
                    panic(err)
                }
                wg.Done()
            }(path)
        }
        return nil
    })
    if err != nil {
        panic(err)
    }
    wg.Wait()
}

func md5File(filePath string) (string, error) {
    file, err := os.Open(filePath)
    if err != nil {
        return "", err
    }
    defer file.Close()

    hash := md5.New()
    if _, err := io.Copy(hash, file); err != nil {
        return "", err
    }

    checksum := hash.Sum(nil)

    return string(hex.EncodeToString(checksum)), nil
}

func writeToFile(filePath, md5sum, output string) error {

    file, err := os.OpenFile(output, os.O_WRONLY|os.O_APPEND|os.O_CREATE, 0755)
    if err != nil {
        return err
    }

    defer file.Close()

    file.WriteString(fmt.Sprintf("%s %s\n", md5sum, filePath))

    return file.Sync()
}

根据我的理解,在写入输出文件时,有时肯定会遇到竞争条件,但它从来没有遇到过。我的意思是我已经执行了这段代码无数次,从未遇到任何问题。它甚至每次都会产生相同的结果。

这是为什么呢?我错过了什么吗?

更新:当我说它必然会面临竞争条件时,我的意思是当运行多个 goroutine 时,多个 goroutine 可能会同时写入文件。 p>

解决方案


当我说它必然会面临竞争条件时,我的意思是当运行多个 goroutine 时,多个 goroutine 可能会同时写入文件。

多次打开一个文件不是问题。由于您显式使用 O_APPEND ,因此写入之间不会产生严重交互。引用 来自man open

O_APPEND ...
在每次 write(2) 之前,文件偏移量位于文件末尾,就像 lseek(2) 一样。 文件偏移量的修改和写入操作作为单个原子步骤执行

今天关于《为何这段代码没有满足并发要求?》的内容介绍就到此结束,如果有什么疑问或者建议,可以在golang学习网公众号下多多回复交流;文中若有不正之处,也希望回复留言以告知!

声明:本文转载于:stackoverflow 如有侵犯,请联系study_golang@163.com删除
相关阅读
更多>
最新阅读
更多>
课程推荐
更多>