GO實(shí)現(xiàn)文件上傳操作
本文實(shí)例為大家分享了GO實(shí)現(xiàn)文件上傳操作的具體代碼,供大家參考,具體內(nèi)容如下
由于需求中有文件上傳這一個(gè)需求,在這里我們就學(xué)習(xí)一下go語(yǔ)言如何上傳文件。本文主要通過(guò)表單的方式進(jìn)行文件上傳操作。主要有以下三步:
- 表單中增加
enctype
屬性 - 服務(wù)端調(diào)用
r.ParseMultipartForm
,把上傳的文件存儲(chǔ)在內(nèi)存和臨時(shí)文件中 - 使用
r.FormFile
獲取文件句柄,然后對(duì)文件進(jìn)行存儲(chǔ)等處理。
1、表單操作
要使表單能夠上傳文件,首先第一步就要添加form的enctype
屬性進(jìn)去,enctype
屬性有如下三種情況:
application/x-www-form-urlencoded 表示在發(fā)送前編碼所有字符(默認(rèn))
multipart/form-data 不對(duì)字符編碼。在使用包含文件上傳控件的表單時(shí),必須使用該值。
text/plain 空格轉(zhuǎn)換為 "+" 加號(hào),但不對(duì)特殊字符編碼。
所以可以創(chuàng)建如下上傳表單:
<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、服務(wù)端操作
在服務(wù)端只需要添加一個(gè)handlerFunc
并完善相關(guān)功能即可:
// 處理/upload 邏輯 func upload(w http.ResponseWriter, r *http.Request) { ? ? //獲取請(qǐng)求的方法 ?? ?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 { ?? ??? ?//設(shè)置內(nèi)存大小 ?? ??? ?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) ?? ??? ?//創(chuàng)建上傳目錄 ?? ??? ?os.Mkdir("./test", os.ModePerm) ?? ??? ?//創(chuàng)建上傳文件 ?? ??? ?f, err := os.Create("./test/" + handler.Filename) ?? ??? ?//f, err := os.OpenFile("./test/"+handler.Filename, os.O_WRONLY|os.O_CREATE, 0666) // 此處假設(shè)當(dāng)前目錄下已存在test目錄 ?? ??? ?if err != nil { ?? ??? ??? ?fmt.Println(err) ?? ??? ??? ?return ?? ??? ?} ?? ??? ?defer f.Close() ?? ??? ?io.Copy(f, file) ?? ?} }
在main()
函數(shù)中記得添加http.HandleFunc("/upload", upload)
即可。
通過(guò)http://127.0.0.1:9999/upload來(lái)測(cè)試文件上傳。 截圖
選擇文件之后就會(huì)在當(dāng)前目錄下的test文件夾中成功上傳文件。
3、流程解析
通過(guò)上面的代碼可以看到,處理文件上傳我們需要調(diào)用r.ParseMultipartForm
,里面的參數(shù)表示maxMemory
,調(diào)用ParseMultipartForm
之后,上傳的文件存儲(chǔ)在maxMemory
大小的內(nèi)存里面,如果文件大小超過(guò)了maxMemory
,那么剩下的部分將存儲(chǔ)在系統(tǒng)的臨時(shí)文件中。我們可以通過(guò)r.FormFile
獲取上面的文件句柄,然后實(shí)例中使用了io.Copy
來(lái)存儲(chǔ)文件。我們可以嘗試使用看一下使用的相關(guān)原函數(shù):
ParseMultipartForm
函數(shù)如下:
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
函數(shù)如下:
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
里面的結(jié)構(gòu)體如下
// 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、成功結(jié)果
瀏覽器端顯示如下消息。
以上就是本文的全部?jī)?nèi)容,希望對(duì)大家的學(xué)習(xí)有所幫助,也希望大家多多支持腳本之家。
- GO語(yǔ)言原生實(shí)現(xiàn)文件上傳功能
- Go語(yǔ)言實(shí)現(xiàn)文件上傳
- Go實(shí)現(xiàn)文件分片上傳
- Go Gin實(shí)現(xiàn)文件上傳下載的示例代碼
- golang實(shí)現(xiàn)ftp實(shí)時(shí)傳輸文件的案例
- Golang實(shí)現(xiàn)異步上傳文件支持進(jìn)度條查詢的方法
- golang socket斷點(diǎn)續(xù)傳大文件的實(shí)現(xiàn)方法
- 用go gin server來(lái)做文件上傳服務(wù)
- Golang+Android基于HttpURLConnection實(shí)現(xiàn)的文件上傳功能示例
- Golang實(shí)現(xiàn)文件傳輸功能
相關(guān)文章
Go語(yǔ)言超時(shí)退出的三種實(shí)現(xiàn)方式總結(jié)
這篇文章主要為大家詳細(xì)介紹了Go語(yǔ)言中超時(shí)退出的三種實(shí)現(xiàn)方式,文中的示例代碼簡(jiǎn)潔易懂,對(duì)我們深入了解Go語(yǔ)言有一定的幫助,需要的可以了解一下2023-06-06GoLang實(shí)現(xiàn)Viper庫(kù)的封裝流程詳解
Viper是一個(gè)用于Go語(yǔ)言應(yīng)用程序的配置管理庫(kù),它提供了一種簡(jiǎn)單而靈活的方式來(lái)處理應(yīng)用程序的配置,支持多種格式的配置文件,這篇文章主要介紹了GoLang封裝Viper庫(kù)的流程,感興趣的同學(xué)可以參考下文2023-05-05golang中使用proto3協(xié)議導(dǎo)致的空值字段不顯示的問題處理方案
這篇文章主要介紹了golang中使用proto3協(xié)議導(dǎo)致的空值字段不顯示的問題處理方案,文中通過(guò)示例代碼介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友們下面隨著小編來(lái)一起學(xué)習(xí)學(xué)習(xí)吧2019-10-10Go語(yǔ)言狀態(tài)機(jī)的實(shí)現(xiàn)
本文主要介紹了Go語(yǔ)言狀態(tài)機(jī)的實(shí)現(xiàn),文中通過(guò)示例代碼介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友們下面隨著小編來(lái)一起學(xué)習(xí)學(xué)習(xí)吧2023-03-03golang通過(guò)gorm操作sqlite設(shè)置主鍵自增的步驟
這篇文章主要介紹了golang通過(guò)gorm操作sqlite設(shè)置主鍵自增的詳細(xì)步驟,本文給大家介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或工作具有一定的參考借鑒價(jià)值,需要的朋友可以參考下2023-09-09go mod 安裝依賴 unkown revision問題的解決方案
這篇文章主要介紹了go mod 安裝依賴 unkown revision問題的解決方案,具有很好的參考價(jià)值,希望對(duì)大家有所幫助。一起跟隨小編過(guò)來(lái)看看吧2021-05-05詳解Go語(yǔ)言如何利用上下文進(jìn)行并發(fā)計(jì)算
在Go編程中,上下文(context)是一個(gè)非常重要的概念,它包含了與請(qǐng)求相關(guān)的信息,本文主要來(lái)和大家討論一下如何在并發(fā)計(jì)算中使用上下文,感興趣的可以了解下2024-02-02