简介
这个系列将使用 Golang 模仿 PHP 的 Http 客户端之大作 Guzzle 。虽然 Http Client 的封装不是什么难事,甚至不用封装也可以写出能跑的代码,但是有一个封装良好,使用简单,得心应手的好工具能让代码更具优雅,更健壮,更易读,Guzzle 就是 PHP 里面我见过为数不少的 Http Client,很多 Composer 包都直接使用 Guzzle 发请求,你还在用 curl 吗?试试 Guzzle 吧!
我接触 Golang 其实已经很久了,从 2017 年刚大学毕业那会就在自己看书看教程学习了,因为工作中用的是 PHP 语言,算上平时花在其他地方的时间,Go 的学习时间其实也不长,自己也没信心找 Golang 的工作。但经验告诉我,语言是有相通性的,凌驾在语言之上的知识也是相通的,所以我打算多码一码~
Server 基础
要写 Client ,我觉得必须先熟悉 Server ,先看下 Go Http Server 是怎么玩的?
启动一个 Http Server
在 Go 里面,启动一个 Http Server 非常简单,就像这样:
package main
import (
"net/http"
"log"
)
func main() {
mux := http.NewServerMux()
mux.handleFunc("/",index)
err := http.ListenAndServe(":8888",mux)
if err != nil {
log.Fatal(err)
}
}
func index(w http.ResponseWriter,r *http.Request){
fmt.Fprintf(w,"hello,golang")
}
err := http.ListenAndServe(":8888",mux)
index 这个方法是 http handler ,就是处理请求的方法,通过 mux 注册之后,有请求过来,go 就会找到对应的 handler 执行。其中注册的方式有很多种,这里列出的只是其中一种,感兴趣的可以去了解其他方式~
接收 query string
通过 get 参数的形式传递到后端,获取方式有两种,一是取 r.URL.Query() 里面的,二是使用 r.Form 。(这里的 r 指的是 handler 接收参数里面的 *http.Request)
func handleQueryString(w http.ResponseWriter,r *http.Request) {
v := r.URL.Query()
fmt.Println(v.Get("age"))
r.ParseForm()
fmt.Println(r.Form["likes"])
fmt.Println(r.Form.Get("likes"))
fmt.Println(r.Form.Get("age"))
bytes,err := json.Marshal(r.Form)
if err != nil {
fmt.Fprintln(w,err.Error())
return
}
fmt.Fprintln(w,string(bytes))
}
用 postman 测试看看,控制台输出:
可以看到 r.URL.Qury() 和 r.Form 都拿到了 query string ,但是对于“数组”这种特殊类型,go 似乎并没有帮我们封装好,在 php 里面,这个 likes 后端到时候用 $_GET 获取到的就是一个数组,key 是 "likes" ,value 是包含“篮球”+“足球”的数组,不过问题不大,这里 go 对于每个值都用了 slice ,所以我只需要传递的时候用相同的 key ,就能达到一样的效果。
这里额外说明,r.Form["k"] 这种形式拿到的是 slice ,r.Form.GET("k") 这种形式拿到的是 string ,所以用单值还是多值完全由我们自己决定。
接收 post 数据(x-www-form-urlencoded)
接收 post 数据,依然可以使用 r.Form 。但是仅限 enctype=x-www-form-urlencoded 的情况,r.Form 同时包含了 post 数据和 query string ,如果要区分,可以使用 r.PostForm 只取 post 里面的参数。
接收 post 表单数据(form-data)
func handleFormData(w http.ResponseWriter,r *http.Request) {
err := r.ParseMultipartForm(1024)
if err != nil {
fmt.Fprintln(w,err.Error())
return
}
bytes,err := json.Marshal(r.MultipartForm.Value)
if err != nil {
fmt.Fprintln(w,err.Error())
return
}
fmt.Fprintln(w,string(bytes))
}
可以发现,r.MultipartForm.Value 里面有我们想要的东西,但是注意到 pic 这张图片貌似没有拿到,莫慌,继续看~
接收 post 文件(form-data)
func handleFile(w http.ResponseWriter,r *http.Request) {
err := r.ParseMultipartForm(1024)
if err != nil {
fmt.Fprintln(w,err.Error())
return
}
bytes,err := json.Marshal(r.MultipartForm.File)
if err != nil {
fmt.Fprintln(w,err.Error())
return
}
fmt.Fprintln(w,string(bytes))
}
完美,r.MultipartForm.File 里面有我们想要的。
接收 post json 数据(application/json)
func handleJson(w http.ResponseWriter,r *http.Request) {
data,err := ioutil.ReadAll(r.Body)
if err != nil {
fmt.Fprintf(w,err.Error())
return
}
fmt.Fprintf(w,string(data))
}
这个是藏在 r.Body 里面,Read 出来是 []byte 类型,意味着并不只是 json ,其他类型如:application/xml,plain/text 都用这个接收。
总结
服务端接收数据的大部分场景已经覆盖,还有 Cookie Session 什么的,留在后面 Client 出来后再补上。
Next Step:Client 快速封装。
2021-02-10