網站首頁 編程語言 正文
本文實例為大家分享了GO實現文件上傳操作的具體代碼,供大家參考,具體內容如下
由于需求中有文件上傳這一個需求,在這里我們就學習一下go語言如何上傳文件。本文主要通過表單的方式進行文件上傳操作。主要有以下三步:
- 表單中增加
enctype
屬性 - 服務端調用
r.ParseMultipartForm
,把上傳的文件存儲在內存和臨時文件中 - 使用
r.FormFile
獲取文件句柄,然后對文件進行存儲等處理。
1、表單操作
要使表單能夠上傳文件,首先第一步就要添加form的enctype
屬性進去,enctype
屬性有如下三種情況:
application/x-www-form-urlencoded ? 表示在發送前編碼所有字符(默認)
multipart/form-data ? ? ?不對字符編碼。在使用包含文件上傳控件的表單時,必須使用該值。
text/plain ? ? ?空格轉換為 "+" 加號,但不對特殊字符編碼。
所以可以創建如下上傳表單:
<html> <head> ? ? <title>上傳文件</title> </head> <body> <form enctype="multipart/form-data" action="/upload" method="post"> ? <input type="file" name="uploadfile" /> ? <input type="hidden" name="token" value="{{.}}"/> ? <input type="submit" value="upload" /> </form> </body> </html>
2、服務端操作
在服務端只需要添加一個handlerFunc
并完善相關功能即可:
// 處理/upload 邏輯 func upload(w http.ResponseWriter, r *http.Request) { ? ? //獲取請求的方法 ?? ?fmt.Println("method:", r.Method) ? ? //GET的處理操作 ?? ?if r.Method == "GET" { ?? ??? ?crutime := time.Now().Unix() ?? ??? ?h := md5.New() ?? ??? ?io.WriteString(h, strconv.FormatInt(crutime, 10)) ?? ??? ?token := fmt.Sprintf("%x", h.Sum(nil)) ?? ??? ?t, _ := template.ParseFiles("upload.gtpl") ?? ??? ?t.Execute(w, token) ?? ?} else { ?? ??? ?//設置內存大小 ?? ??? ?r.ParseMultipartForm(32 << 20) ?? ??? ?//獲取上傳文件 ?? ??? ?file, handler, err := r.FormFile("uploadfile") ?? ??? ?if err != nil { ?? ??? ??? ?fmt.Println(err) ?? ??? ??? ?return ?? ??? ?} ?? ??? ?defer file.Close() ?? ??? ?fmt.Fprintf(w, "%v", handler.Header) ?? ??? ?//創建上傳目錄 ?? ??? ?os.Mkdir("./test", os.ModePerm) ?? ??? ?//創建上傳文件 ?? ??? ?f, err := os.Create("./test/" + handler.Filename) ?? ??? ?//f, err := os.OpenFile("./test/"+handler.Filename, os.O_WRONLY|os.O_CREATE, 0666) // 此處假設當前目錄下已存在test目錄 ?? ??? ?if err != nil { ?? ??? ??? ?fmt.Println(err) ?? ??? ??? ?return ?? ??? ?} ?? ??? ?defer f.Close() ?? ??? ?io.Copy(f, file) ?? ?} }
在main()
函數中記得添加http.HandleFunc("/upload", upload)
即可。
通過http://127.0.0.1:9999/upload來測試文件上傳。 截圖
選擇文件之后就會在當前目錄下的test文件夾中成功上傳文件。
3、流程解析
通過上面的代碼可以看到,處理文件上傳我們需要調用r.ParseMultipartForm
,里面的參數表示maxMemory
,調用ParseMultipartForm
之后,上傳的文件存儲在maxMemory
大小的內存里面,如果文件大小超過了maxMemory
,那么剩下的部分將存儲在系統的臨時文件中。我們可以通過r.FormFile
獲取上面的文件句柄,然后實例中使用了io.Copy
來存儲文件。我們可以嘗試使用看一下使用的相關原函數:
ParseMultipartForm
函數如下:
func (r *Request) ParseMultipartForm(maxMemory int64) error { ?? ?if r.MultipartForm == multipartByReader { ?? ??? ?return errors.New("http: multipart handled by MultipartReader") ?? ?} ?? ?if r.Form == nil { ?? ??? ?err := r.ParseForm() ?? ??? ?if err != nil { ?? ??? ??? ?return err ?? ??? ?} ?? ?} ?? ?if r.MultipartForm != nil { ?? ??? ?return nil ?? ?} ?? ?mr, err := r.multipartReader(false) ?? ?if err != nil { ?? ??? ?return err ?? ?} ?? ?f, err := mr.ReadForm(maxMemory) ?? ?if err != nil { ?? ??? ?return err ?? ?} ?? ?if r.PostForm == nil { ?? ??? ?r.PostForm = make(url.Values) ?? ?} ?? ?for k, v := range f.Value { ?? ??? ?r.Form[k] = append(r.Form[k], v...) ?? ??? ?// r.PostForm should also be populated. See Issue 9305. ?? ??? ?r.PostForm[k] = append(r.PostForm[k], v...) ?? ?} ?? ?r.MultipartForm = f ?? ?return nil }
FormFile
函數如下:
func (r *Request) FormFile(key string) (multipart.File, *multipart.FileHeader, error) { ?? ?if r.MultipartForm == multipartByReader { ?? ??? ?return nil, nil, errors.New("http: multipart handled by MultipartReader") ?? ?} ?? ?if r.MultipartForm == nil { ?? ??? ?err := r.ParseMultipartForm(defaultMaxMemory) ?? ??? ?if err != nil { ?? ??? ??? ?return nil, nil, err ?? ??? ?} ?? ?} ?? ?if r.MultipartForm != nil && r.MultipartForm.File != nil { ?? ??? ?if fhs := r.MultipartForm.File[key]; len(fhs) > 0 { ?? ??? ??? ?f, err := fhs[0].Open() ?? ??? ??? ?return f, fhs[0], err ?? ??? ?} ?? ?} ?? ?return nil, nil, ErrMissingFile }
文件handler
是multipart.FileHeader
里面的結構體如下
// A FileHeader describes a file part of a multipart request. type FileHeader struct { ?? ?Filename string ?? ?Header ? textproto.MIMEHeader ?? ?Size ? ? int64 ?? ?content []byte ?? ?tmpfile string }
4、成功結果
瀏覽器端顯示如下消息。
原文鏈接:https://blog.csdn.net/qq_40185499/article/details/104124046
相關推薦
- 2022-10-07 numpy拼接矩陣的實現_python
- 2023-08-01 前端傳遞對象數組,后端使用list接收并解析
- 2022-07-13 RedisTemplate無法根據key 獲取值的問題
- 2022-04-30 Python語言中的if語句詳情_python
- 2022-07-21 ubutu虛擬機和主機共享同一網絡
- 2022-07-07 關于C++智能指針shared_ptr和unique_ptr能否互轉問題_C 語言
- 2023-07-07 electron 運行以及打包超時解決方法
- 2023-02-12 完美解決Redis在雙擊redis-server.exe出現閃退問題_Redis
- 最近更新
-
- 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同步修改后的遠程分支