Go文件下载需设Content-Type和Content-Disposition头,用io.Copy流式传输,校验文件名防路径遍历,并用filename*=UTF-8''支持中文名。
在 Go 中处理文件下载,核心是通过 http.ResponseWriter 正确设置响应头并写入文件内容(即输出文件流)。关键不是“保存文件”,而是让浏览器识别为可下载资源并触发下载行为。
浏览器是否弹出下载对话框,主要取决于两个响应头:
application/octet-stream(通用二进制流),避免因 MIME 类型被浏览器尝试内嵌打开(如 PDF、图片);若明确类型且希望强制下载,也可用具体类型(如 application/pdf)+ 强制 disposition。attachment; filename="xxx.ext",其中 filename 建议使用 ASCII 字符,中文名需用 filename*=UTF-8''xxx.ext 编码(兼容性更好)。不要一次性将大文件全部读入内存。推荐用 io.Copy 直接从文件句柄复制到 ResponseWriter,底层自动分块传输:
os.Open 打开文件,记得 defer f.Close();w.Header().Set() 设置上述两个关键 header;io.Copy(w, f) 流式传输,高效且内存友好;os.Open 错误(如文件不存在、权限不足),返回 404 或 403 状态码。用户请求的文件名若来自 URL 参数(如 /download?name=报告.pdf),务必做白名单校验或路径清理,防止目录遍历攻击(如 ../../../etc/passwd):
filepath.Base() 提取纯文件名,再结合白名单验证;Content-Disposition 推荐写成:attachment; filename="report.pdf"; filename*=UTF-8''%E6%8A%A5%E5%91%8A.pdf
以下是一个安全、流式、支持中文名的下载 handler 示例:
func downloadHandler(w http.ResponseWriter, r *http.Request) {
filename := r.URL.Query().Get("name")
if filename == "" {
http.Error(w, "missing name", http.StatusBadRequest)
return
}
// 安全过滤:只保留合法字符,禁止路径遍历
cleanName := filepath.Base(filename)
if !isValidFilename(cleanName) {
http.Error(w, "invalid filename", http.StatusBadRequest)
return
}
filePath := "./uploads/" + cleanName
f, err := os.Open(filePath)
if os.IsNotExist(err) {
http.Error(w, "file not found", http.StatusNotFound)
return
}
if err != nil {
http.Error(w, "cannot open
file", http.StatusInternalServerError)
return
}
defer f.Close()
// 获取文件真实大小用于 Content-Length(可选但推荐)
fi, _ := f.Stat()
w.Header().Set("Content-Length", fmt.Sprintf("%d", fi.Size()))
// 设置强制下载 + 支持中文名
encodedName := url.PathEscape(cleanName) // 简单转义(生产建议用 mime.BEncoding)
w.Header().Set("Content-Type", "application/octet-stream")
w.Header().Set("Content-Disposition", fmt.Sprintf(`attachment; filename="%s"; filename*=UTF-8''%s`, cleanName, encodedName))
// 流式输出
io.Copy(w, f)
}
func isValidFilename(name string) bool {
match, _ := regexp.MatchString(`^[a-zA-Z0-9\u4e00-\u9fa5._\-]+\.([a-zA-Z0-9]{2,})$`, name)
return match
}
基本上就这些。不复杂但容易忽略 header 设置和路径安全 —— 少一步,就可能变成文件被浏览器打开、乱码,甚至服务器文件被任意读取。
# go
# golang
# 编码
# 浏览器
# app
# usb
# pdf
# stream
# 状态码
# copy
# ASCII
# http
# 流式
# 遍历
# 设为
# 是一个
# 句柄
# 下划线
# 扩展名
# 弹出
# 不存在
# 可选
相关文章:
如何在香港服务器上快速搭建免备案网站?
宝塔面板如何快速创建新站点?
如何通过商城免费建站系统源码自定义网站主题?
html制作网站的步骤有哪些,iapp如何添加网页?
如何使用Golang table-driven基准测试_多组数据测量函数效率
建站之星Pro快速搭建教程:模板选择与功能配置指南
如何通过远程VPS快速搭建个人网站?
Python多线程使用规范_线程安全解析【教程】
成都网站制作价格表,现在成都广电的单独网络宽带有多少的,资费是什么情况呢?
建站主机核心功能解析:服务器选择与网站搭建流程指南
微信推文制作网站有哪些,怎么做微信推文,急?
如何选购建站域名与空间?自助平台全解析
如何通过宝塔面板实现本地网站访问?
电商平台网站制作流程,电商网站如何制作?
建站与域名管理如何高效结合?
广平建站公司哪家专业可靠?如何选择?
网站制作公司排行榜,抖音怎样做个人官方网站
已有域名建站全流程解析:网站搭建步骤与建站工具选择
行程制作网站有哪些,第三方机票电子行程单怎么开?
建站之星如何快速更换网站模板?
建站主机是否属于云主机类型?
建站之星安装后如何配置SEO及设计样式?
网站制作多少钱一个,建一个论坛网站大约需要多少钱?
大连 网站制作,大连天途有线官网?
如何在Ubuntu系统下快速搭建WordPress个人网站?
建站主机如何选?性能与价格怎样平衡?
建站之星代理如何优化在线客服效率?
如何配置WinSCP新建站点的密钥验证步骤?
建站之星导航配置指南:自助建站与SEO优化全解析
如何在自有机房高效搭建专业网站?
ppt在线制作免费网站推荐,有什么下载免费的ppt模板网站?
郑州企业网站制作公司,郑州招聘网站有哪些?
如何用搬瓦工VPS快速搭建个人网站?
重庆网站制作公司哪家好,重庆中考招生办官方网站?
青岛网站建设如何选择本地服务器?
攀枝花网站建设,攀枝花营业执照网上怎么年审?
,sp开头的版面叫什么?
电影网站制作价格表,那些提供免费电影的网站,他们是怎么盈利的?
常州企业网站制作公司,全国继续教育网怎么登录?
深圳网站制作设计招聘,关于服装设计的流行趋势,哪里的资料比较全面?
建站之星2.7模板:企业网站建设与h5定制设计专题
如何快速搭建个人网站并优化SEO?
建站之星后台密码如何安全设置与找回?
,网页ppt怎么弄成自己的ppt?
,石家庄四十八中学官网?
头像制作网站在线制作软件,dw网页背景图像怎么设置?
Android自定义listview布局实现上拉加载下拉刷新功能
常州自助建站:操作简便模板丰富,企业个人快速搭建网站
盐城做公司网站,江苏电子版退休证办理流程?
网站制作免费,什么网站能看正片电影?
*请认真填写需求信息,我们会在24小时内与您取得联系。