ASP.NET Core – Upload and Download Files (上传和下载文件)
前言
以前得文章 Asp.net core 学习笔记 ( upload/download files 文件上传与下载 ), 这篇是修订版.
Handle Upload File (处理上传文件)
我的例子是基于 Web API 和 Ajax 的.
前端 Ajax
const btn = document.querySelector('button')!;
btn.addEventListener('click', async () => {
const formData = new FormData();
formData.append('file', createTextBlob('Hello World'), 'text1.txt');
// formData.append('file', createTextBlob('Derrick'), 'text2.txt'); // for multiple file
await fetch('http://localhost:5221/Upload', {
method: 'POST',
body: formData,
});
function createTextBlob(text: string): Blob {
const textEncoder = new TextEncoder();
const bytes = textEncoder.encode(text);
return new Blob([bytes], {
type: 'text/plain',
});
}
});
upload 一个/多个 text file.
Web API
DTO
public class UploadDto
{
public IFormFile File { get; set; } = null!;
}
关键就是这个 IFormFile, 它就是用来接收前端 File 的.
Controller
[ApiController]
public class UploadController : ControllerBase
{
[HttpPost("Upload")]
[Consumes("multipart/form-data")]
[ProducesResponseType(StatusCodes.Status200OK)]
public async Task<ActionResult> Upload([FromForm] UploadDto dto)
{
return Ok("ok");
}
}
把 upload file 写入 disk
var fileRootFullPath = Path.Combine(AppContext.BaseDirectory, $@"..\..\..\{dto.File.FileName}");
using var fileStream = System.IO.File.Create(fileRootFullPath);
await dto.File.CopyToAsync(fileStream);
IFormFile 用法很简单, 它通过 .FileName 拿到文件名, 通过 CopyToAsync 把 stream 传给别人. 比如上面是把 stream 导入 file stream, 这样就写入磁盘了.
把 upload file stream 传给任何人使用
// 直接读 string 出来
using var stream = dto.File.OpenReadStream();
using var streamReader = new StreamReader(stream, encoding: Encoding.UTF8);
var text = await streamReader.ReadToEndAsync(); // Hello World
通过 OpenReadStream() 导出 stream 就可以做任何 stream 处理了.
handle multiple files
// JavaScript
const formData = new FormData();
formData.append('file', createTextBlob('Hello World'), 'text1.txt');
formData.append('file', createTextBlob('Derrick'), 'text2.txt'); // for multiple file // C#
public class UploadDto
{
public List<IFormFile> File { get; set; } = null!;
}
没什么提别的, 就只是换成了 List<IFormFile>
file size limit
ASP.NET Core 默认的 max upload file size 是 28.6 MB。
在 Program.cs
修改 Kestrel
builder.WebHost.ConfigureKestrel((context, serverOptions) =>
{
serverOptions.Limits.MaxRequestBodySize = 157286400; // 150MB
});
修改 FormOptions
builder.Services.Configure<FormOptions>(options =>
{
options.MultipartBodyLengthLimit = 157286400; // 150 MB
});
Production IIS 的话,修改 web.config
<?xml version="1.0" encoding="utf-8"?>
<configuration>
<system.webServer>
<security>
<requestFiltering>
<!-- Set the maximum allowed content length to 150 MB (in bytes) -->
<requestLimits maxAllowedContentLength="157286400" />
</requestFiltering>
</security>
<!-- Add other configurations as needed -->
</system.webServer>
</configuration>
Download File (下载文件)
前端 HTML
<a href="http://localhost:5221/Download" download="">download</a>
Web API Controller
[ApiController]
public class UploadController : ControllerBase
{
[HttpGet("Download")]
[Produces(MediaTypeNames.Text.Plain)]
[ProducesResponseType(StatusCodes.Status200OK)]
public ActionResult Download()
{
return new FileContentResult(Encoding.UTF8.GetBytes("Hello World"), "text/plain")
{
FileDownloadName = "text.txt"
}; // 下面这个写法和上面是等价的
// return File(
// fileContents: Encoding.UTF8.GetBytes("Hello World"),
// contentType: "text/plain",
// fileDownloadName: "text.txt"
// );
}
}
返回 bytes, 声明 ContentType 和 Download File Name 就可以了. 简单
Force open save as dialog (showSaveFilePicker)
用 Chrome 下载文件的体验是,它会直接放到 default location folder 里,除非用户开启 “Ask where to save each file before downloading” 设置

Firefox 也是这样。无意间发现 pixlr.com 在 Chrome 可以 force open save as dialog。
我猜它用了一个比较新的 File System API : showSaveFilePicker。目前只有 Chrome 支持。
参考: Stack Overflow – Force showing the "Save as" dialog box when downloading a file
代码大致长这样
const canvas = document.querySelector<HTMLCanvasElement>('.img')!;
const ctx = canvas.getContext('2d')!;
const response = await fetch('https://192.168.1.152:4200/src/images/nana.jpg');
const blob = await response.blob();
const img = await createImageBitmap(blob);
[canvas.width, canvas.height] = [img.width, img.height];
ctx.drawImage(img, 0, 0);
const btn = document.querySelector<HTMLElement>('.btn')!;
btn.addEventListener('click', () => {
canvas.toBlob(async blob => {
// 关键:
const handle = await showSaveFilePicker({
suggestedName: 'nana.jpg',
types: [
{
description: 'Jpeg file',
accept: {
'image/jpeg': ['.jpeg', '.jpg'],
},
},
],
});
const writable = await handle.createWritable();
await writable.write(blob!);
writable.close();
}, 'image/jpeg');
});
不过这个方式下载的文件,不会出现在 Chrome 的 recently download 记入里面,体验有点差。
另外,目前 TypeScript 还不支持哦,需要
yarn add @types/wicg-file-system-access --dev
ASP.NET Core – Upload and Download Files (上传和下载文件)的更多相关文章
- MVC文件上传03-使用Request.Files上传多个文件
本篇体验在控制器方法中使用controllerContext.HttpContext.Request.Files上传多个文件.兄弟篇为: MVC文件上传01-使用jquery异步上传并客户端验证类型和 ...
- Spring Boot之 Controller 接收参数和返回数据总结(包括上传、下载文件)
一.接收参数(postman发送) 1.form表单 @RequestParam("name") String name 会把传递过来的Form表单中的name对应 ...
- 利用SecureCRT上传、下载文件(使用sz与rz命令),超实用!
利用SecureCRT上传.下载文件(使用sz与rz命令),超实用! 文章来源:http://blog.csdn.net/dongqinliuzi/article/details/39623169 借 ...
- 每天一个linux命令(26)--用SecureCRT来上传和下载文件
用SSH管理Linux 服务器时经常需要远程与本地之间交互文件,而直接使用 SecureCRT 自带的上传下载功能无疑是最方便的,SecureCRT下的文件传输协议有ASCII.Xmodem.Zmod ...
- oracle EBS上传和下载文件(转)
最近一直在做一个工作流的项目,最终用户要求在发送消息的时候可以附带附件,这个又是给我的一个难题.在网上查了一下ORACLE上传资料,找到了黄建华前辈写的<Oracle EBS Forms开发指南 ...
- 【Loadrunner】使用LoadRunner上传及下载文件
使用LoadRunner上传及下载文件 1)LoadRunner上传文件 web_submit_data("importStudent.do", "Action=http ...
- SecureCRT上传和下载文件
SecureCRT上传和下载文件(下载默认目录) SecureCR 下的文件传输协议有ASCII .Xmodem .Ymodem .Zmodem ASCII:这是最快的传输协议,但只能传送文本文件. ...
- 11、只允许在主目录下上传和下载文件,不允许用putty登录
创建用户xiao, 使其只允许在用户主目录 (/var/www/html)下上传和下载文件,不允许用putty登录 (为了安全起见,不给过多的权限) 1.创建xiao用户 [root@localh ...
- 每天一个linux命令(26):用SecureCRT来上传和下载文件
用SSH管理linux服务器时经常需要远程与本地之间交互文件.而直接用SecureCRT自带的上传下载功能无疑是最方便的,SecureCRT下的文件传输协议有ASCII.Xmodem.Zmodem. ...
- Linux--用SecureCRT来上传和下载文件
SecureCRT下的文件传输协议有以下几种:ASCII.Xmodem.Ymodem.Zmodem ASCII:这是最快的传输协议,但只能传送文本文件. Xmodem:这种古老的传输协议速度较慢,但由 ...
随机推荐
- Django使用 DoesNotExist 异常和 Logger 来记录异常情况
代码不仅处理了特定的异常类型,还可以添加更多的调试信息来帮助诊断问题.可以使用 DoesNotExist 异常和 Logger 来记录异常情况. from django.core.exceptions ...
- [oeasy]python0101_尾声_PC_wintel_8080_诸神的黄昏_arm_riscv
尾声 回忆上次内容 回顾了 ibm 使用开放架构 用 pc兼容机 战胜了 dec 小型机 apple 个人电脑 触击牺牲打 也破掉了 自己 软硬一体全自主的 金身 借助了 各种 软硬件厂商的 力量 最 ...
- GitHub Star 数量前 12 的开源无代码工具
相关文章:GitHub Star 数量前 15 的开源低代码项目 在本篇文章中,我们将探索 12 款在 GitHub 上星级排名前列的开源无代码工具. 每款工具都旨在简化和加速开发过程,但各自侧重于不 ...
- [rCore学习笔记 07]移除标准库依赖
改造Rust hello world 移除println!宏 rustc添加对裸机的支持 rustup target add riscv64gc-unknown-none-elf detail rus ...
- c++17 using继承所有构造函数
//使用using继承所有的构造函数 #include "tmp.h" #include <iostream> using namespace std; struct ...
- 全网最好看的单细胞umap图绘制教程
全网最好看的单细胞umap图绘制教程 作者按 大家或许都曾被Nature, Science上的单细胞umap图吸引过,不免心生崇拜.在这里,我们将介绍一种简单方便的顶刊级umap图可视化 全文字数|预 ...
- springboot中事务失效的一些场景以及如何应对
@Transactional是基于AOP的,因此事务发生需要两个条件: 1.添加@Transactional注解 2.使用代理对象 失效场景:同一个类中直接调用的类方法,而被调方法带有@Transac ...
- 【Uni-App】page.json 配置项一栏笔记
官方文档 https://uniapp.dcloud.io/collocation/pages 一些配置项是全局的,也可以在页面对象中设置 { "pages": [ //pages ...
- 关于工业AI辅助制造(模具设计、模样生产制造环节)
关于工业AI辅助制造(模具设计.模样生产制造环节) AI技术的具体使用场景: AI辅助模具设计: AI辅助模具安装工艺参数调整. 具体方案设想: AI辅助模具设计: 使用AI大模型对历史已有的设计方案 ...
- 强化学习中经验池的替代设计——A3C算法
读论文<Asynchronous methods for deep reinforcement learning>有感 ---------------------------------- ...