Golang流和读者

时间:2018-04-09 12:38:26

标签: go stream tar

我正在编写一个简单的脚本来下载解压缩tar.gz文件,然后将其删除。每当我尝试删除它时,我都会收到错误消息: The process cannot access the file because it is being used by another process.

我认为错误在于我如何将文件传递给extractTarGz函数,但我不确定。

以下是代码:

package main

import (
    "archive/tar"
    "compress/gzip"
    "io"
    "log"
    "os"
)

func main() {
    f, err := os.Open("file.tar.gz")
    if err != nil {
        panic(err)
    }
    defer f.Close()
    extractTarGz(f)

    err = os.Remove("file.tar.gz")
}

func extractTarGz(gzipStream io.Reader) {
    uncompressedStream, err := gzip.NewReader(gzipStream)
    if err != nil {
        log.Fatal("ExtractTarGz: NewReader failed")
    }

    tarReader := tar.NewReader(uncompressedStream)

    for true {
        header, err := tarReader.Next()

        if err == io.EOF {
            break
        }

        if err != nil {
            log.Fatalf("ExtractTarGz: Next() failed: %s", err.Error())
        }

        switch header.Typeflag {
        case tar.TypeDir:
            if err := os.Mkdir(header.Name, 0755); err != nil {
                log.Fatalf("ExtractTarGz: Mkdir() failed: %s", err.Error())
            }
        case tar.TypeReg:
            outFile, err := os.Create(header.Name)
            if err != nil {
                log.Fatalf("ExtractTarGz: Create() failed: %s", err.Error())
            }
            defer outFile.Close()
            if _, err := io.Copy(outFile, tarReader); err != nil {
                log.Fatalf("ExtractTarGz: Copy() failed: %s", err.Error())
            }
        default:
            log.Fatalf(
                "ExtractTarGz: uknown type: %s in %s",
                header.Typeflag,
                header.Name)
        }
    }
}

2 个答案:

答案 0 :(得分:4)

您应首先关闭该文件,然后尝试将其删除。由于您使用defer关闭它,因此会在os.Remove()来电后调用。

试试这样:

name := "file.tar.gz"
defer func() {
    if err = os.Remove(name); err != nil {
        log.Printf("Failed to remove %s: %v", name, err)
    }
}()

f, err := os.Open(name)
if err != nil {
    panic(err)
}
defer f.Close()

extractTarGz(f)

延迟函数以LIFO(后进先出)顺序执行,因此将调用第一个f.Close(),然后尝试删除该文件的另一个apostrophe-email。引自Spec: Deferred statements:

  

...在周围函数返回之前立即调用延迟函数,以相反的顺序调用

答案 1 :(得分:2)

f, err := os.Open("file.tar.gz")
if err != nil {
    panic(err)
}
defer f.Close()
extractTarGz(f)
err = os.Remove("file.tar.gz")

至少,您需要在删除之前关闭文件。

err = f.Close()
if err != nil {
    panic(err)
}
err = os.Remove("file.tar.gz")

defer f.Close()直到函数结束才会运行。