欧美bbbwbbbw肥妇,免费乱码人妻系列日韩,一级黄片

解決golang post文件時Content-Type出現(xiàn)的問題

 更新時間:2021年05月01日 14:03:34   作者:九江Mgx  
這篇文章主要介紹了解決golang post文件時Content-Type出現(xiàn)的問題,具有很好的參考價值,希望對大家有所幫助。一起跟隨小編過來看看吧

同事用php寫了一個接口,要上傳文件,讓我做下測試,直接用curl命令調(diào)用成功,然后想用golang寫個示例,

源碼如下:

package main 
import (
    "bytes" 
    "fmt" 
    "io/ioutil" 
    "mime/multipart" 
    "net/http" 
)
 
func main() { 
    uri := "http://xxxxxxxxxxxx/api/fileattr" //URL地址 xxxxxxxxxxxx由商務(wù)提供 
    name := "xxxxxxxxxxxx" //用戶名 
    pass := "xxxxxxxxxxxx" //密碼 
    fn := "xxxxxxxxxxxx.txt" //文件路徑
 
    //讀出文本文件數(shù)據(jù) 
    file_data, _ := ioutil.ReadFile(fn) 
    body := new(bytes.Buffer) 
    w := multipart.NewWriter(body)
 
    //取出內(nèi)容類型 
    content_type := w.FormDataContentType() 
    //將文件數(shù)據(jù)寫入 
    pa, _ := w.CreateFormFile("file", fn) 
    pa.Write(file_data) 
    //設(shè)置用戶名密碼 
    w.WriteField("name", name) 
    w.WriteField("pass", pass) 
    w.Close() 
    //開始提交
 
    req, _ := http.NewRequest("POST", uri, body) 
    req.Header.Set("Content-Type", content_type) 
    resp, _ := http.DefaultClient.Do(req) 
    data, _ := ioutil.ReadAll(resp.Body) 
    resp.Body.Close() 
    fmt.Println(resp.StatusCode) 
    fmt.Printf("%s", data) 
}

發(fā)現(xiàn)總是調(diào)用失敗,返回文件類型不對,詢問后得知,同事做了判斷,文件只能為text/plain類型,抓包發(fā)現(xiàn),我提交時的文件類型為:application/octet-stream,仔細(xì)查看golang源碼:mime/multipart/write.go,CreateFormFile的源碼是這樣的:

func (w *Writer) CreateFormFile(fieldname, filename string) (io.Writer, error) { 
    h := make(textproto.MIMEHeader) 
    h.Set("Content-Disposition", 
        fmt.Sprintf(`form-data; name="%s"; filename="%s"`, 
            escapeQuotes(fieldname), escapeQuotes(filename))) 
    h.Set("Content-Type", "application/octet-stream") 
    return w.CreatePart(h) 
}

可以得知Content-Type被固定為了application/octet-stream,知道原因了,問題就好解決了。

第一種方法

就是直接修改CreateFormFile,或者加個CreateFormFile2命令,這種方法將來golang升級后可能會出問題。

第二種方法

可以自己來CreatePart:

h := make(textproto.MIMEHeader)
    h.Set("Content-Disposition",
        fmt.Sprintf(`form-data; name="%s"; filename="%s"`,
            escapeQuotes(fieldname), escapeQuotes(filename)))
    h.Set("Content-Type", "text/plain")

再用 w.CreatePart(h)得到io.Writer,問題解決!這種方法不侵入golang源代碼,最終代碼如下:

package main 
import (
    "bytes"
    "fmt"
    "io/ioutil"
    "mime/multipart"
    "net/http"
    "net/textproto"
)
 
func main() {
    uri := "http://xxxxxxxxxxxx/api/fileattr" //URL地址 xxxxxxxxxxxx由商務(wù)提供
    name := "xxxxxxxxxx"                      //用戶名
    pass := "xxxxxxx"                         //密碼
    fn := "x:/xxx/xxx.txt"                    //文件路徑
 
    //讀出文本文件數(shù)據(jù)
    file_data, _ := ioutil.ReadFile(fn)
 
    body := new(bytes.Buffer)
    w := multipart.NewWriter(body)
 
    //取出內(nèi)容類型
    content_type := w.FormDataContentType()
 
    //將文件數(shù)據(jù)寫入
    h := make(textproto.MIMEHeader)
    h.Set("Content-Disposition",
        fmt.Sprintf(`form-data; name="%s"; filename="%s"`,
            "file", //參數(shù)名為file
            fn))
    h.Set("Content-Type", "text/plain") //設(shè)置文件格式
    pa, _ := w.CreatePart(h)
    pa.Write(file_data)
 
    //設(shè)置用戶名密碼
    w.WriteField("name", name)
    w.WriteField("pass", pass)
 
    w.Close() 
    //開始提交
    req, _ := http.NewRequest("POST", uri, body)
    req.Header.Set("Content-Type", content_type)
    resp, _ := http.DefaultClient.Do(req)
    data, _ := ioutil.ReadAll(resp.Body)
    resp.Body.Close()
    fmt.Println(resp.StatusCode)
    fmt.Printf("%s", data)
}

補(bǔ)充:用go來玩最簡單的web服務(wù)器------順便說說Content-Type字段

web服務(wù)端代碼s.go:

package main 
import (
    "io"
    "log"
    "net/http"
)
 
func handlerHello(w http.ResponseWriter, r *http.Request) {
    io.WriteString(w, "hello girls")
}
 
func main() {
    http.HandleFunc("/hello", handlerHello)     // 注冊   
    err := http.ListenAndServe("localhost:8080", nil)
    if err != nil {
        log.Println(err)
    }
}

go run s.go一下,跑起來, 然后在瀏覽器執(zhí)行http://127.0.0.1:8080/hello (或者在命令行用curl發(fā)http請求也可以), 瀏覽器上的結(jié)果為:

hello girls

好簡單??梢栽诳蛻舳嘶蛘叻?wù)端抓包看下, 很典型的http req和rsp.

我們再來看一個有趣的問題, 修改s.go為:

package main 
import (
    "io"
    "log"
    "net/http"
)
 
func handlerHello(w http.ResponseWriter, r *http.Request) {
    str := `
        table border="1">
        <tr>
        <td>row 1, cell 1</td>
        <td>row 1, cell 2</td>
        </tr>
        <tr>
        <td>row 2, cell 1</td>
        <td>row 2, cell 2</td>
        </tr>
        </table>
        ` 
    io.WriteString(w, str)
}
 
func main() {
    http.HandleFunc("/hello", handlerHello)     // 注冊   
    err := http.ListenAndServe("localhost:8080", nil)
    if err != nil {
        log.Println(err)
    }
}

再次重啟服務(wù)并發(fā)請求, 瀏覽器上顯示的內(nèi)容是:

table border="1">
 <tr>
	 <td>row 1, cell 1</td>
	 <td>row 1, cell 2</td>
 </tr>
 <tr>
	 <td>row 2, cell 1</td>
	 <td>row 2, cell 2</td>
 </tr>
</table>

抓包看一下, 發(fā)現(xiàn)有:Content-Type: text/plain; charset=utf-8

因此, 瀏覽器需要根據(jù)純文本顯示。 注意到, 上述的table左邊少了一個"<". 我們加上后,

s.go的代碼如下:

package main 
import (
    "io"
    "log"
    "net/http"
)
 
func handlerHello(w http.ResponseWriter, r *http.Request) {
    str := `
        <table border="1">
        <tr>
        <td>row 1, cell 1</td>
        <td>row 1, cell 2</td>
        </tr>
        <tr>
        <td>row 2, cell 1</td>
        <td>row 2, cell 2</td>
        </tr>
        </table>
        ` 
    io.WriteString(w, str)
}
 
func main() {
    http.HandleFunc("/hello", handlerHello)     // 注冊   
    err := http.ListenAndServe("localhost:8080", nil)
    if err != nil {
        log.Println(err)
    }
}

再次重啟服務(wù),發(fā)請求,瀏覽器端的顯示是:

row 1, cell 1 row 1, cell 2
row 2, cell 1 row 2, cell 2

抓包看, 有Content-Type: text/html; charset=utf-8

可見, 服務(wù)端會判斷str的格式,來確定Content-Type的類型, 從而決定了瀏覽器端的展示方式。服務(wù)端的自動判斷行為, 有點意思。 在我看來, 這樣不太好,應(yīng)該讓程序員來指定Content-Type.

以上為個人經(jīng)驗,希望能給大家一個參考,也希望大家多多支持腳本之家。如有錯誤或未考慮完全的地方,望不吝賜教。

相關(guān)文章

  • Go 1.13中errors包的新變化示例解析

    Go 1.13中errors包的新變化示例解析

    這篇文章主要為大家介紹了Go 1.13中errors包的新變化示例解析,有需要的朋友可以借鑒參考下,希望能夠有所幫助,祝大家多多進(jìn)步,早日升職加薪
    2023-12-12
  • Windows下安裝VScode 并使用及中文配置方法

    Windows下安裝VScode 并使用及中文配置方法

    這篇文章主要介紹了Windows下安裝VScode 并使用及中文配置的方法詳解,本文通過圖文并茂的形式給大家介紹,對大家的學(xué)習(xí)或工作具有一定的參考借鑒價值,需要的朋友可以參考下
    2020-03-03
  • 使用 Go 管理版本的方法示例

    使用 Go 管理版本的方法示例

    這篇文章主要介紹了使用 Go 管理版本的方法示例,文中通過示例代碼介紹的非常詳細(xì),對大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價值,需要的朋友們下面隨著小編來一起學(xué)習(xí)學(xué)習(xí)吧
    2019-10-10
  • 詳解如何在Golang中監(jiān)聽多個channel

    詳解如何在Golang中監(jiān)聽多個channel

    這篇文章主要為大家詳細(xì)介紹了如何在Golang中實現(xiàn)監(jiān)聽多個channel,文中的示例代碼講解詳細(xì),感興趣的小伙伴可以跟隨小編一起學(xué)習(xí)一下
    2023-03-03
  • Golang實現(xiàn)字符串倒序的幾種解決方案

    Golang實現(xiàn)字符串倒序的幾種解決方案

    給定一個字符串,按單詞將該字符串逆序是我們大家在開發(fā)中可能會遇到的一個需求,所以下面這篇文章主要給大家介紹了關(guān)于Golang如何實現(xiàn)字符串倒序的幾種解決方案,文中通過示例代碼介紹的非常詳細(xì),需要的朋友可以參考借鑒,下面來一起看看吧。
    2017-10-10
  • go?variant底層原理深入解析

    go?variant底層原理深入解析

    這篇文章主要為大家介紹了go?variant底層原理深入解析,有需要的朋友可以借鑒參考下,希望能夠有所幫助,祝大家多多進(jìn)步,早日升職加薪
    2022-11-11
  • Go Resiliency庫中timeout實現(xiàn)原理及源碼解析

    Go Resiliency庫中timeout實現(xiàn)原理及源碼解析

    Go-Resiliency庫中的timeout是一種基于協(xié)程的超時機(jī)制,通過創(chuàng)建協(xié)程來執(zhí)行任務(wù)并設(shè)置超時時間,若任務(wù)執(zhí)行時間超時則中止協(xié)程并返回錯誤,需要詳細(xì)了解可以參考下文
    2023-05-05
  • Go 函數(shù)返回nil遇到問題避坑分析

    Go 函數(shù)返回nil遇到問題避坑分析

    這篇文章主要為大家介紹了Go 函數(shù)返回nil遇到的避坑問題分析,有需要的朋友可以借鑒參考下,希望能夠有所幫助,祝大家多多進(jìn)步,早日升職加薪
    2023-01-01
  • 總結(jié)Go語言中defer的使用和注意要點

    總結(jié)Go語言中defer的使用和注意要點

    Go語言中的defer關(guān)鍵字實現(xiàn)比較特殊的功能,這篇文章給大家總結(jié)了關(guān)于Go語言中defer的使用和注意要點,有需要的朋友們可以參考借鑒,下面來一起看看吧。
    2016-09-09
  • golang并發(fā)編程中Goroutine 協(xié)程的實現(xiàn)

    golang并發(fā)編程中Goroutine 協(xié)程的實現(xiàn)

    Go語言中的協(xié)程是一種輕量級線程,通過在函數(shù)前加go關(guān)鍵字來并發(fā)執(zhí)行,具有動態(tài)棧、快速啟動和低內(nèi)存使用等特點,本文就來詳細(xì)的介紹一下,感興趣的可以了解一下
    2024-10-10

最新評論