Go语言如何实现文件上传

本文实例为大家分享了Go实现文件上传的具体代码,供大家参考,具体内容如下 文件上传:客户端把上传文件转换为二进制流后发送给服务器,服务器对二进制流进行解析 HT

本文实例为大家分享了Go实现文件上传的具体代码,供大家参考,具体内容如下

文件上传:客户端把上传文件转换为二进制流后发送给服务器,服务器对二进制流进行解析

HTML表单(form)enctype(Encode Type)属性控制表单在提交数据到服务器时数据的编码类型.

enctype=”application/x-www-form-urlencoded” 默认值,表单数据会被编码为名称/值形式

  • enctype=”multipart/form-data” 编码成消息,每个控件对应消息的一部分.请求方式必须是post
  • enctype=”text/plain” 纯文本形式进行编码的

HTML模版内容如下(在项目/view/index.html)

<!DOCTYPE html>
<html lang="en">
<head>
    <meta charset="UTF-8">
    <title>文件上传</title>
</head>
<body>
<form action="upload" enctype="multipart/form-data" method="post">
    用户名:<input type="text" name="username"/><br/>
    密码:<input type="file" name="photo"/><br/>
    <input type="submit" value="注册"/>
</form>
</body>
</html>

服务端可以使用FormFIle(“name”)获取上传到的文件,官方定义如下

// FormFile returns the first file for the provided form key.
// FormFile calls ParseMultipartForm and ParseForm if necessary.
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
}

multipart.File是文件对象

// File is an interface to access the file part of a multipart message.
// Its contents may be either stored in memory or on disk.
// If stored on disk, the File's underlying concrete type will be an *os.File.
type File interface {
    io.Reader
    io.ReaderAt
    io.Seeker
    io.Closer
}

封装了文件的基本信息

// A FileHeader describes a file part of a multipart request.
type FileHeader struct {
    Filename string                    //文件名
    Header   textproto.MIMEHeader    //MIME信息
    Size     int64                    //文件大小,单位bit

    content []byte                    //文件内容,类型[]byte
    tmpfile string                    //临时文件
}

服务器端编写代码如下

  • 获取客户端传递后的文件流,把文件保存到服务器即可
package main

import (
   "net/http"
   "fmt"
   "html/template"
   "io/ioutil"
)

/*
显示欢迎页upload.html
 */
func welcome(rw http.ResponseWriter, r *http.Request) {
   t, _ := template.ParseFiles("template/html/upload.html")
   t.Execute(rw, nil)
}

/*
文件上传
 */
func upload(rw http.ResponseWriter, r *http.Request) {
   //获取普通表单数据
   username := r.FormValue("username")
   fmt.Println(username)
   //获取文件流,第三个返回值是错误对象
   file, header, _ := r.FormFile("photo")
   //读取文件流为[]byte
   b, _ := ioutil.ReadAll(file)
   //把文件保存到指定位置
   ioutil.WriteFile("D:/new.png", b, 0777)
   //输出上传时文件名
   fmt.Println("上传文件名:", header.Filename)
}

func main() {
   server := http.Server{Addr: "localhost:8899"}

   http.HandleFunc("/", welcome)
   http.HandleFunc("/upload", upload)

   server.ListenAndServe()
}

以上就是本文的全部内容,希望对大家的学习有所帮助,也希望大家多多支持好代码网。

您可能有感兴趣的文章
golang并发编程的如何实现

golang如何实现redis的延时消息队列功能示例

golang中range在slice和map遍历中的注意事项

Golang命令行进行debug调试操作

golang的空标识符理解