21

我正在尝试创建一个包含一个或多个文件的 zip 文件。
我正在使用 .NET 框架 4.5,更具体地说是 System.IO.Compression 命名空间。
目标是允许用户通过 ASP.NET MVC 应用程序下载 zip 文件。
正在生成 zip 文件并将其发送到客户端,但是当我尝试通过双击打开它时,我收到以下错误:
Windows 无法打开文件夹。压缩(压缩)文件夹...无效。
这是我的代码:

[HttpGet]
public FileResult Download()
{
    var fileOne = CreateFile(VegieType.POTATO);
    var fileTwo = CreateFile(VegieType.ONION);
    var fileThree = CreateFile(VegieType.CARROT);

    IEnumerable<FileContentResult> files = new List<FileContentResult>() { fileOne, fileTwo, fileThree };
    var zip = CreateZip(files);

    return zip;
}

private FileContentResult CreateFile(VegieType vType)
{
    string fileName = string.Empty;
    string fileContent = string.Empty;

    switch (vType)
    {
        case VegieType.BATATA:
            fileName = "batata.csv";
            fileContent = "THIS,IS,A,POTATO";
            break;
        case VegieType.CEBOLA:
            fileName = "cebola.csv";
            fileContent = "THIS,IS,AN,ONION";
            break;
        case VegieType.CENOURA:
            fileName = "cenoura.csv";
            fileContent = "THIS,IS,A,CARROT";
            break;
        default:
            break;
    }

    var fileBytes = Encoding.GetEncoding(1252).GetBytes(fileContent);
    return File(fileBytes, MediaTypeNames.Application.Octet, fileName);
}

private FileResult CreateZip(IEnumerable<FileContentResult> files)
{
    byte[] retVal = null;

    if (files.Any())
    {
        using (MemoryStream zipStream = new MemoryStream())
        {
            using (ZipArchive archive = new ZipArchive(zipStream, ZipArchiveMode.Create, false))
            {
                foreach (var f in files)
                {
                    var entry = archive.CreateEntry(f.FileDownloadName, CompressionLevel.Fastest);
                    using (var entryStream = entry.Open())
                    {
                        entryStream.Write(f.FileContents, 0, f.FileContents.Length);
                        entryStream.Close();
                    }
                }

                zipStream.Position = 0;
                retVal = zipStream.ToArray();
            }
        }
    }

    return File(retVal, MediaTypeNames.Application.Zip, "horta.zip");
}

任何人都可以解释一下为什么当我双击它时,Windows 会说我的 zip 文件无效。
最后考虑一下,我可以使用 7-Zip 打开它。

4

4 回答 4

33

在 ZipArchive 对象被释放后,您需要通过 ToArray 获取 MemoryStream 缓冲区。否则,您最终会得到损坏的存档。

请注意,我已更改 ZipArchive 构造函数的参数以在添加条目时保持打开状态。

在处理 ZipArchive 时会进行一些校验和,因此如果您之前阅读过 MemoryStream,它仍然不完整。

    private FileResult CreateZip(IEnumerable<FileContentResult> files)
    {
        byte[] retVal = null;

        if (files.Any())
        {
            using (MemoryStream zipStream = new MemoryStream())
            {
                using (ZipArchive archive = new ZipArchive(zipStream, ZipArchiveMode.Create, true))
                {
                    foreach (var f in files)
                    {
                        var entry = archive.CreateEntry(f.FileDownloadName, CompressionLevel.Fastest);
                        using (BinaryWriter writer = new BinaryWriter(entry.Open()))
                        {                                   
                            writer.Write(f.FileContents, 0, f.FileContents.Length);
                            writer.Close();
                        }
                    }

                    zipStream.Position = 0;
                }
                retVal = zipStream.ToArray();
            }
        }

        return File(retVal, MediaTypeNames.Application.Zip, "horta.zip");
    }
于 2016-10-21T13:14:18.313 回答
3

只需返回流...

private ActionResult CreateZip(IEnumerable files)
{
    if (files.Any())
    {
        MemoryStream zipStream = new MemoryStream();
        using (ZipArchive archive = new ZipArchive(zipStream, ZipArchiveMode.Create, false))
        {
            foreach (var f in files)
            {
               var entry = archive.CreateEntry(f.FileDownloadName, CompressionLevel.Fastest);
               using (var entryStream = entry.Open())
               {
                   entryStream.Write(f.FileContents, 0, f.FileContents.Length);
                   entryStream.Close();
               }
           }

        }

        zipStream.Position = 0;
        return File(zipStream, MediaTypeNames.Application.Zip, "horta.zip");
    }

    return new EmptyResult();
}
于 2016-10-21T13:25:12.337 回答
1

尝试改变

using (ZipArchive archive = new ZipArchive(zipStream, ZipArchiveMode.Create, false))

using (ZipArchive archive = new ZipArchive(zipStream, ZipArchiveMode.Create, true))

在这种用法中,存档在关闭时被强制写入流。但是,如果构造函数的leaveOpen参数设置为 false,它也会关闭底层流。

于 2018-02-21T13:11:47.993 回答
0

当我在示例中为条目添加了错误的名称时

var fileToZip = "/abc.txt";
ZipArchiveEntry zipFileEntry = zipArchive.CreateEntry(fileToZip);

我得到了同样的错误。更正文件名后,现在可以了。

于 2019-04-08T10:55:39.157 回答