網站首頁 編程語言 正文
一般來說,通過c.Request.FormFile()獲取文件的時候,所有內容都全部讀到了內存。如果是個巨大的文件,則可能內存會爆掉;且,有的時候我們需要一邊上傳一邊處理。
以下的代碼實現了大文件流式上傳。
還非常不完美,但是可以作為參考:
upload.html
<!DOCTYPE html>
<html lang="en">
<head>
<meta charset="UTF-8">
<title>upload file</title>
</head>
<body>
<form method="post" enctype="multipart/form-data" action="/gin_upload">
<input type="file" name="ff" multiple="multiple"/><br/>
<input type="submit" value="提交"/>
</form>
</body>
gin_stream_upload_file.go
/* 本例子實現了gin框架下的多個大文件流式上傳,避免了文件內容存在內存而無法支持大文件的情況 */ package main import ( "fmt" "github.com/gin-gonic/gin" "os" "bytes" "io" "log" "strconv" "strings" ) /// 解析多個文件上傳中,每個具體的文件的信息 type FileHeader struct{ ContentDisposition string Name string FileName string ///< 文件名 ContentType string ContentLength int64 } /// 解析描述文件信息的頭部 /// @return FileHeader 文件名等信息的結構體 /// @return bool 解析成功還是失敗 func ParseFileHeader(h []byte) (FileHeader, bool){ arr := bytes.Split(h, []byte("\r\n")) var out_header FileHeader out_header.ContentLength = -1 const ( CONTENT_DISPOSITION = "Content-Disposition: " NAME = "name=\"" FILENAME = "filename=\"" CONTENT_TYPE = "Content-Type: " CONTENT_LENGTH = "Content-Length: " ) for _,item := range arr{ if bytes.HasPrefix(item, []byte(CONTENT_DISPOSITION)){ l := len(CONTENT_DISPOSITION) arr1 := bytes.Split(item[l:], []byte("; ")) out_header.ContentDisposition = string(arr1[0]) if bytes.HasPrefix(arr1[1], []byte(NAME)){ out_header.Name = string(arr1[1][len(NAME):len(arr1[1])-1]) } l = len(arr1[2]) if bytes.HasPrefix(arr1[2], []byte(FILENAME)) && arr1[2][l-1]==0x22{ out_header.FileName = string(arr1[2][len(FILENAME):l-1]) } } else if bytes.HasPrefix(item, []byte(CONTENT_TYPE)){ l := len(CONTENT_TYPE) out_header.ContentType = string(item[l:]) } else if bytes.HasPrefix(item, []byte(CONTENT_LENGTH)){ l := len(CONTENT_LENGTH) s := string(item[l:]) content_length,err := strconv.ParseInt(s, 10, 64) if err!=nil{ log.Printf("content length error:%s", string(item)) return out_header, false } else { out_header.ContentLength = content_length } } else { log.Printf("unknown:%s\n", string(item)) } } if len(out_header.FileName)==0{ return out_header,false } return out_header,true } /// 從流中一直讀到文件的末位 /// @return []byte 沒有寫到文件且又屬于下一個文件的數據 /// @return bool 是否已經讀到流的末位了 /// @return error 是否發生錯誤 func ReadToBoundary(boundary []byte, stream io.ReadCloser, target io.WriteCloser)([]byte, bool, error){ read_data := make([]byte, 1024*8) read_data_len := 0 buf := make([]byte, 1024*4) b_len := len(boundary) reach_end := false for ;!reach_end; { read_len, err := stream.Read(buf) if err != nil { if err != io.EOF && read_len<=0 { return nil, true, err } reach_end = true } //todo: 下面這一句很蠢,值得優化 copy(read_data[read_data_len:], buf[:read_len]) //追加到另一塊buffer,僅僅只是為了搜索方便 read_data_len += read_len if (read_data_len<b_len+4){ continue } loc := bytes.Index(read_data[:read_data_len], boundary) if loc>=0{ //找到了結束位置 target.Write(read_data[:loc-4]) return read_data[loc:read_data_len],reach_end, nil } target.Write(read_data[:read_data_len-b_len-4]) copy(read_data[0:], read_data[read_data_len-b_len-4:]) read_data_len = b_len + 4 } target.Write(read_data[:read_data_len]) return nil, reach_end, nil } /// 解析表單的頭部 /// @param read_data 已經從流中讀到的數據 /// @param read_total 已經從流中讀到的數據長度 /// @param boundary 表單的分割字符串 /// @param stream 輸入流 /// @return FileHeader 文件名等信息頭 /// []byte 已經從流中讀到的部分 /// error 是否發生錯誤 func ParseFromHead(read_data []byte, read_total int, boundary []byte, stream io.ReadCloser)(FileHeader, []byte, error){ buf := make([]byte, 1024*4) found_boundary := false boundary_loc := -1 var file_header FileHeader for { read_len, err := stream.Read(buf) if err!=nil{ if err!=io.EOF{ return file_header, nil, err } break } if read_total+read_len>cap(read_data){ return file_header, nil, fmt.Errorf("not found boundary") } copy(read_data[read_total:], buf[:read_len]) read_total += read_len if !found_boundary { boundary_loc = bytes.Index(read_data[:read_total], boundary) if -1 == boundary_loc { continue } found_boundary = true } start_loc := boundary_loc+len(boundary) file_head_loc := bytes.Index(read_data[start_loc:read_total], []byte("\r\n\r\n")) if -1==file_head_loc{ continue } file_head_loc += start_loc ret := false file_header,ret = ParseFileHeader(read_data[start_loc:file_head_loc]) if !ret{ return file_header,nil,fmt.Errorf("ParseFileHeader fail:%s", string(read_data[start_loc:file_head_loc])) } return file_header, read_data[file_head_loc+4:read_total], nil } return file_header,nil,fmt.Errorf("reach to sream EOF") } func main(){ log.SetFlags(log.LstdFlags | log.Lshortfile) r := gin.Default() r.StaticFile("/upload.html", "./upload.html") r.POST("/gin_upload", func(c *gin.Context) { var content_length int64 content_length = c.Request.ContentLength if content_length<=0 || content_length>1024*1024*1024*2{ log.Printf("content_length error\n") return } content_type_,has_key := c.Request.Header["Content-Type"] if !has_key{ log.Printf("Content-Type error\n") return } if len(content_type_)!=1{ log.Printf("Content-Type count error\n") return } content_type := content_type_[0] const BOUNDARY string = "; boundary=" loc := strings.Index(content_type, BOUNDARY) if -1==loc{ log.Printf("Content-Type error, no boundary\n") return } boundary := []byte(content_type[(loc+len(BOUNDARY)):]) log.Printf("[%s]\n\n", boundary) // read_data := make([]byte, 1024*12) var read_total int = 0 for { file_header, file_data, err := ParseFromHead(read_data, read_total, append(boundary, []byte("\r\n")...), c.Request.Body) if err != nil { log.Printf("%v", err) return } log.Printf("file :%s\n", file_header.FileName) // f, err := os.Create(file_header.FileName) if err != nil { log.Printf("create file fail:%v\n", err) return } f.Write(file_data) file_data = nil //需要反復搜索boundary temp_data, reach_end, err := ReadToBoundary(boundary, c.Request.Body, f) f.Close() if err != nil { log.Printf("%v\n", err) return } if reach_end{ break } else { copy(read_data[0:], temp_data) read_total = len(temp_data) continue } } // c.JSON(200, gin.H{ "message": fmt.Sprintf("%s", "ok"), }) }) r.Run() }
原文鏈接:https://www.cnblogs.com/ahfuzhang/p/12629416.html#5077708
相關推薦
- 2022-05-11 SpringBoot整合RabbitMq與高級特性
- 2022-09-06 C語言超詳細講解指向函數的指針_C 語言
- 2022-10-26 Golang?手寫一個簡單的并發任務?manager_Golang
- 2022-04-03 ?Python?代碼制作動態鞭炮_python
- 2022-11-25 Centos?8.2?升級內核通過elrepo源的方法_云其它
- 2022-04-16 ASP.NET?Core命令行界面CLI用法_基礎應用
- 2022-08-05 利用jQuery?treetable實現樹形表格拖拽詳解_jquery
- 2023-02-18 C++中std::thread線程用法_C 語言
- 最近更新
-
- window11 系統安裝 yarn
- 超詳細win安裝深度學習環境2025年最新版(
- Linux 中運行的top命令 怎么退出?
- MySQL 中decimal 的用法? 存儲小
- get 、set 、toString 方法的使
- @Resource和 @Autowired注解
- Java基礎操作-- 運算符,流程控制 Flo
- 1. Int 和Integer 的區別,Jav
- spring @retryable不生效的一種
- Spring Security之認證信息的處理
- Spring Security之認證過濾器
- Spring Security概述快速入門
- Spring Security之配置體系
- 【SpringBoot】SpringCache
- Spring Security之基于方法配置權
- redisson分布式鎖中waittime的設
- maven:解決release錯誤:Artif
- restTemplate使用總結
- Spring Security之安全異常處理
- MybatisPlus優雅實現加密?
- Spring ioc容器與Bean的生命周期。
- 【探索SpringCloud】服務發現-Nac
- Spring Security之基于HttpR
- Redis 底層數據結構-簡單動態字符串(SD
- arthas操作spring被代理目標對象命令
- Spring中的單例模式應用詳解
- 聊聊消息隊列,發送消息的4種方式
- bootspring第三方資源配置管理
- GIT同步修改后的遠程分支