使用 ASP.NET Core 创建和下载 zip 文件

        对于最近的一个功能,我必须从用 ASP.NET Core 编写的内部网站下载一批文件。在下载文件之前对其进行压缩,结果证明这是一种轻松实现多文件下载的好方法。.NET 提供了所有需要的功能,在本文中,我将向您展示如何实现它。

        首先,我将创建一个新的 ASP.NET Core 网站: 

dotnet new mvc 

        我选择了 MVC 模板,但是没有任何与 zip 相关的代码是特定于 MVC 的。

        在本例中,我将创建一个能够压缩和下载一些文件的端点。在现实生活中,后端通常需要输入参数才能知道要压缩什么,但为了简单起见,我将省略这部分。

        首先声明没有主体的方法:

[Route("downloadzip")]
public async Task<IActionResult> DownloadTheZipFile()
{
    // ...

        代码尚未编译,因此让我们修复它。首先构建要压缩的文件列表。在下面的代码中,我将硬编码一些路径,但每个文件可能来自客户端、数据库或第三方:

List<string> files = new List<string>
{
    "first/file.txt",
    "second/file.txt"
};

        接下来,我们需要邮政编码:

using (var memoryStream = new MemoryStream())
{
    using (var zipArchive = new ZipArchive(memoryStream, ZipArchiveMode.Create, true))
    {
        foreach (var file in files)
        {
            zipArchive.CreateEntryFromFile(file, Path.GetFileName(file));
        }
    }
}

        代码使用ZipArchive.NET 中提供的类来创建 zip 文件。它被包装在 中,MemoryStream因为我们想从方法中返回一个文件流:

memoryStream.Position = 0;
return File(memoryStream, "application/zip", "download.zip");

        重置流后,我将其作为File-method 的一部分返回。

整个方法如下:

[Route("downloadzip")]
public async Task<IActionResult> DownloadTheZipFile()
{
    List<string> files = new List<string>
    {
        "first/file.txt",
        "second/file.txt"
    };

    using (var memoryStream = new MemoryStream())
    {
        using (var zipArchive = new ZipArchive(memoryStream, ZipArchiveMode.Create, true))
        {
            foreach (var file in files)
            {
                zipArchive.CreateEntryFromFile(file, Path.GetFileName(file));
            }
        }
        
        memoryStream.Position = 0;
        return File(memoryStream, "application/zip", "my.zip");
    }
}

点击F5并调用/downloadzip端点来见证奇迹的发生。

        本文中的示例非常简单,没有考虑任何问题。如果您要处理大型 zip 文件,将 zip 文件写入服务器上的临时文件,然后将文件流式传输到客户端可能会更有效。这可以帮助防止因尝试一次将整个 zip 文件保存在内存中而可能出现的内存问题:

var tempFile = Path.GetTempFileName();

using (var zipFile = System.IO.File.Create(tempFile))
using (var zipArchive = new ZipArchive(zipFile, ZipArchiveMode.Create))
{
    foreach (var file in files)
    {
        zipArchive.CreateEntryFromFile(file, Path.GetFileName(file));
    }
}

var stream = new FileStream(tempFile, FileMode.Open);
return File(stream, "application/zip", "my.zip");

        另外,请记住,压缩和下载大文件可能需要一些时间。在客户端上实现某种进度可以避免用户尝试多次下载 zip 文件,从而占用额外的服务器资源。

        就是这样。使用内置类,在 .NET 中压缩文件很容易。公平地说,也有一些不错的外部 NuGet 包可用。比如SharpZipLib(我过去曾使用过)和DotNetZip。

如果您喜欢此文章,请收藏、点赞、评论,谢谢,祝您快乐每一天。 

一个基于ASP.NET Core的可伸缩、通用的文件服务器。 通常后端项目可能会有头像、图片、音频、视频等上传/下载需求,这些需求都可以抽象为文件服务。 功能特点 支持Linux(推荐)、Windows 可伸缩式架构,支持部署1-N台文件服务器 RESTful架构的API接口,支持多语言客户端 支持文件秒传、断点续传、远程拉取上传 支持为用户指定磁盘空间配额 支持自定义文件处理器 系统架构 文件的上传/下载通常由客户端直接与文件服务器交互,上传时需要提供代表用户身份token(由业务服务器生成),成功后会返回文件根地址。 也可以直接由业务服务器上传返回文件根地址给客户端。 源码中包含基于.Net Standard的服务端SDK,可以生成token、上传文件等 源码中包含基于.Net Standard的客户端SDK,可以上传/下载文件后端使用 配置业务服务器 //Startup.cs代码片段 public void ConfigureServices(IServiceCollection services) { //.... services.AddFileService(opts => { opts.Host = "fs.mondol.info"; //文件服务器域名 opts.AppSecret = "xxxxxx"; //加密密钥,需要与文件服务器相同 }); } 生成访问令牌 IFileServiceManager fileSvceMgr; //此实例可通过DI框架获得 //根据业务规定其意义,例如:1-代表管理员,2-代表用户 var ownerType = 2; var ownerId = 2; //如果ownerType=2,则为用户ID var validTime = TimeSpan.FromDays(2); //token有效期 var ownerToken = fileSvceMgr.GenerateOwnerTokenString(ownerType, ownerId, validTime); 前端使用 文件上传 IFileServiceClient fileClient; //此实例可通过DI框架获得 var ownerToken = "业务服务器返回的token"; var periodMinute = 0; //有效期,0不过期 var updResult = await fileClient.UploadAsync(ownerToken, "文件路径", periodMinute); var url = updResult.Data.Url; //得到文件根地址 标签:文件服务器
评论
成就一亿技术人!
拼手气红包6.0元
还能输入1000个字符
 
红包 添加红包
表情包 插入表情
 条评论被折叠 查看
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包

打赏作者

hefeng_aspnet

你的鼓励将是我创作的最大动力

¥1 ¥2 ¥4 ¥6 ¥10 ¥20
扫码支付:¥1
获取中
扫码支付

您的余额不足,请更换扫码支付或充值

打赏作者

实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值