信息发布→ 登录 注册 退出

如何在Golang中开发文件上传下载功能_Golang文件上传下载项目实战汇总

发布时间:2025-11-03

点击量:
首先实现文件上传功能,通过http.Request.ParseMultipartForm解析表单,调用request.FormFile获取文件句柄,并用io.Copy将文件保存至服务器路径,同时限制文件大小和类型以防止恶意上传。

在Go语言(Golang)开发中,实现文件的上传与下载是Web服务中的常见需求,比如用户头像上传、附件下载等场景。本文将从基础原理到完整示例,带你一步步实现一个简单但实用的文件上传下载服务。

1. 实现文件上传功能

文件上传通常通过HTTP的multipart/form-data格式提交。Golang的net/http包原生支持解析这种请求。

关键步骤:

  • 使用http.Request.ParseMultipartForm()解析表单数据
  • 调用request.FormFile()获取上传的文件句柄
  • 使用io.Copy()将文件内容保存到服务器指定路径
  • 限制文件大小和类型,防止恶意上传

示例代码:

func uploadHandler(w http.ResponseWriter, r *http.Request) {
    if r.Method != "POST" {
        http.Error(w, "只允许POST请求", http.StatusMethodNotAllowed)
        return
    }

    // 解析 multipart 表单,最大内存 32MB
    err := r.ParseMultipartForm(32 << 20)
    if err != nil {
        http.Error(w, "解析表单失败", http.StatusBadRequest)
        return
    }

    file, handler, err := r.FormFile("file")
    if err != nil {
        http.Error(w, "获取文件失败", http.StatusBadRequest)
        return
    }
    defer file.Close()

    // 创建本地文件
    dst, err := os.Create("./uploads/" + handler.Filename)
    if err != nil {
        http.Error(w, "创建文件失败", http.StatusInternalServerError)
        return
    }
    defer dst.Close()

    // 拷贝文件内容
    _, err = io.Copy(dst, file)
    if err != nil {
        http.Error(w, "保存文件失败", http.StatusInternalServerError)
        return
    }

    fmt.Fprintf(w, "文件 %s 上传成功", handler.Filename)
}

2. 实现文件下载功能

文件下载的核心是设置正确的响应头,告诉浏览器这是一个需要下载的文件,而不是直接展示。

关键点:

  • 使用os.Open()打开服务器上的文件
  • 设置Content-Disposition头触发下载
  • 设置Content-Typeapplication/octet-stream表示二进制流
  • 使用http.ServeFile()io.Copy()发送文件内容

示例代码:

func downloadHandler(w http.ResponseWriter, r *http.Request) {
    filename := r.URL.Query().Get("name")
    if filename == "" {
        http.Error(w, "缺少文件名", http.StatusBadRequest)
        return
    }

    filepath := "./uploads/" + filename

    // 检查文件是否存在
    if _, err := os.Stat(filepath); os.IsNotExist(err) {
        http.Error(w, "文件不存在", http.StatusNotFound)
        return
    }

    // 设置响应头
    w.Header().Set("Content-Disposition", "attachment; filename="+filename)
    w.Header().Set("Content-Type", "application/octet-stream")

    // 发送文件
    http.ServeFile(w, r, filepath)
}

3. 安全与优化建议

在生产环境中,必须考虑安全性和稳定性。

  • 文件名安全:避免路径穿越攻击,如../../etc/passwd。可使用path.Base()提取文件名
  • 文件类型校验:检查文件扩展名或读取前几个字节(magic number)判断真实类型
  • 大小限制:在ParseMultipartForm中设置合理上限
  • 防重命名冲突:可用UUID或时间戳重命名上传文件
  • 并发访问控制:确保多个用户同时上传不会冲突

4. 完整项目结构示例

一个简单的项目目录结构:

/fileserver
├── main.go
├── uploads/
├── static/
│   └── upload.html
└── go.mod

main.go 路由注册:

func main() {
    mux := http.NewServeMux()
    mux.HandleFunc("/upload", uploadHandler)
    mux.HandleFunc("/download", downloadHandler)
    mux.Handle("/", http.FileServer(http.Dir("./static/")))

    fmt.Println("服务启动在 :8080")
    log.Fatal(http.ListenAndServe(":8080", mux))
}

配合一个简单的HTML页面即可测试上传功能。

基本上就这些。掌握以上内容后,你可以在此基础上扩展出支持分片上传、断点续传、权限验证等功能。核心在于理解HTTP文件传输机制和Go的标准库能力。
标签:# go  # golang  # go语言  # html  
在线客服
服务热线

服务热线

4008888355

微信咨询
二维码
返回顶部
×二维码

截屏,微信识别二维码

打开微信

微信号已复制,请打开微信添加咨询详情!