Golang使用Gin框架实现HTTP上传文件过程介绍
作者:中国流浪猫 发布时间:2024-04-28 09:18:12
标签:Go,HTTP,Gin,上传文件
HTTP上传的文件的原理
HTTP协议的文件上传是通过HTTP POST请求实现的,使用multipart/form-data格式将待上传的文件放入请求体中。
服务器根据请求头中的boundary参数来解析请求体,并根据Content-Disposition字段获取文件名等信息,根据Content-Type字段判断文件类型并保存到相应位置。
Gin框架文件上传Demo
代码逻辑:
通过Gin框架封装的Form表单获取数据,获取上传文件
获取文件名,并创建新的文件存储
将上传的文件内容写入新的文件
返回上传成功信息
package main
import (
"fmt"
"github.com/gin-gonic/gin"
"io"
"net/http"
"os"
)
func uploadFile(c *gin.Context) {
//form表单
file, header, err := c.Request.FormFile("upload")
if err != nil {
c.String(http.StatusBadRequest, fmt.Sprintf("上传文件失败: %s", err.Error()))
return
}
// 获取文件名,并创建新的文件存储
filename := header.Filename
out, err := os.Create(filename)
if err != nil {
c.String(http.StatusBadRequest, fmt.Sprintf("创建文件: %s", err.Error()))
return
}
defer out.Close()
//将读取的文件流写到文件中
_, err = io.Copy(out, file)
if err != nil {
c.String(http.StatusBadRequest, fmt.Sprintf("读取文件失败: %s", err.Error()))
return
}
c.String(http.StatusCreated, "上传成功 \n")
}
func main() {
router := gin.Default()
//路由:http://localhost:8080/upload
router.POST("/upload", uploadFile)
router.Run(":8080")
}
限制文件上传的大小
使用 http.MaxBytesReader()
函数来限制 HTTP 请求中读取的最大字节数。这个函数会返回一个新的 Reader 对象,该对象会在读取请求的正文时自动检查字节数,如果超过指定的最大字节数,则会自动停止读取,返回错误。
//限制大小为2M
c.Request.Body = http.MaxBytesReader(c.Writer, c.Request.Body, int64(2<<20))
file, header, err := c.Request.FormFile("upload")
if err != nil {
c.String(http.StatusBadRequest, fmt.Sprintf("上传文件失败: %s", err.Error()))
return
}
该代码不能限制文件上传大小,只是设置内存大小,即使文件大小比这个大,也会写入临时文件
router := gin.Default()
router.MaxMultipartMemory = 2 * 1024 //2M Byte,默认32M
运行结果截图
文件类型验证
验证上传的文件类型,以确保上传的文件是我们期望的类型,借助“github.com/h2non/filetype”实现对文件类型的判断
import (
"fmt"
"github.com/gin-gonic/gin"
"github.com/h2non/filetype"
"io"
"net/http"
)
func uploadFile(c *gin.Context) {
//form表单
c.Request.Body = http.MaxBytesReader(c.Writer, c.Request.Body, int64(2<<20))
file, _, err := c.Request.FormFile("upload")
if err != nil {
c.String(http.StatusBadRequest, fmt.Sprintf("上传文件失败: %s", err.Error()))
return
}
content, err := io.ReadAll(file)
if err != nil {
c.String(http.StatusBadRequest, fmt.Sprintf("读取失败: %s", err.Error()))
return
}
// 解析文件类型
kind, err := filetype.Match(content)
if err != nil {
c.String(http.StatusBadRequest, fmt.Sprintf("文件类型判断失败: %s", err.Error()))
return
}
fmt.Println(kind)
// 验证文件类型
if kind == filetype.Unknown {
c.String(http.StatusCreated, "未知类型 \n")
return
}
if filetype.IsImage(content) {
c.String(http.StatusCreated, "图片 上传成功 \n")
return
}
c.String(http.StatusCreated, "上传成功 \n")
}
文件上传进度-后台计算文件上传进度
实现原理:
要实现 Gin 框架中的文件上传进度,在文件上传中,计算已上传的字节数,并将其与文件的总大小进行比较,以确定上传的进度。
package main
import (
"fmt"
"github.com/gin-gonic/gin"
"net/http"
"os"
)
func uploadFile(c *gin.Context) {
//form表单
//c.Request.Body = http.MaxBytesReader(c.Writer, c.Request.Body, int64(2<<20))
file, fileHeader, err := c.Request.FormFile("upload")
if err != nil {
c.String(http.StatusBadRequest, fmt.Sprintf("上传文件失败: %s", err.Error()))
return
}
filename := fileHeader.Filename
out, err := os.Create(filename)
if err != nil {
c.String(http.StatusBadRequest, fmt.Sprintf("创建文件: %s", err.Error()))
return
}
defer out.Close()
count := 0
for {
buf := make([]byte, 10000)
n, err := file.Read(buf)
if err != nil {
c.String(http.StatusBadRequest, fmt.Sprintf("读取失败: %s", err.Error()))
return
}
if n == 0 {
break
}
count = count + n
out.Write(buf)
fmt.Println(count, float64(fileHeader.Size))
progress := float64(count) / float64(fileHeader.Size) * 100
fmt.Println(fmt.Sprintf("%.2f%%", progress))
}
c.String(http.StatusCreated, "上传成功 \n")
}
func main() {
router := gin.Default()
router.MaxMultipartMemory = 2 * 1024 //2M Byte,默认32M
//路由:http://localhost:8080/upload
router.POST("/upload", uploadFile)
fmt.Println(router.MaxMultipartMemory)
router.Run(":8080")
}
来源:https://blog.csdn.net/a41888313/article/details/129979126


猜你喜欢
- python中实现静态方法和类方法都是依赖于python的修饰器来实现的。 对象方法有self参数,类方法有cls参数,静态方法是不需要这些
- 1.Go语言String的本质就是一个[]byte,所以他们之间可以互相转换,byte数组的长度就是字符串的长度。func StringTe
- 1 包简介1.1 工作空间go语言的工作空间必须由 bin、pkg、src三个目录组成,可以在GOPATH环境变量中添加多个工作空间,但不能
- 遇到的问题:在做爬虫的时候,爬取的url链接内还有转义字符,反斜杠 \,打算用正则的re.sub()替换掉的时候遇到了问题,这是要做替换的字
- 假设访问的views.py如下1.使用url配置默认页from django.conf.urls import urlfrom django
- 第一步:更改setting.py中的DATABASES# 配置数据库DATABASES = { 'default'
- 使用Northwind 数据库首先查询Employees表查询结果:city列里面只有5个城市使用ROW_NUMBER() OVER(PAR
- 脚手架vue-cli 搭建的项目,build后放于服务器上,发现其他资源全部变成静态加载成功,但路由视图为空,最初以为是webpack打包的
- 如下:re.split(pattern, string, [maxsplit], [flags])pattern:表示模式字符串,由要匹配的
- 创建主键可以有两种方式:create table 表名(字段名 类型,字段名 类型,……primary key(name));或者是crea
- 在读文件时常常得到一些\n和引号之类的符号,可以使用字符串的成员函数strip()来去除。1.去除首尾不需要的字符a= '"
- SnowNLP是国人开发的python类库,可以方便的处理中文文本内容,是受到了TextBlob的启发而写的,由于现在大部分的自然语言处理库
- 如下所示:from numpy import *import numpy as npimport matplotlib.pyplot as
- Tensorflow内置了许多数据集,但是实际自己应用的时候还是需要使用自己的数据集,这里TensorFlow 官网也给介绍文档,官方文档。
- 六、XML展望 任何一项新技术的产生都是有其需求背景的,XML的诞生是在HTML遇到不可克服的困难之后。近年来HTML在许多复杂的Web应用
- 本文中介绍了一个MySQL的存储过程,其中涉及Cursor的使用,示例如下:CREATE PROCEDURE `justifyGroupNu
- Python Json读写操作_JsonPath用法详解1. 介绍JSONPath是一种信息抽取类库,是从JSON文档中抽取指定信息的工具,
- 1、查找字符串除了使用index()方法在字符串中查找指定元素,还可以使用find()方法在一个较长的字符串中查找子串。如果找到子串,返回子
- 开始一个组件,毫无目的的写代码是一个不好的习惯,要经历 分析 => 抽象 => 实现 => 应用 四个阶段。组件DEMO地
- 1. 文件的读写原理:文件的读写称为I/O操作。操作原理:.py文件是用解释器去运行,调用OS操作系统的资源,去操作磁盘上的文件。操作流程: