HTTP客户端
基本的HTTP/HTTPS请求
Get、Head、Post和PostForm函数发出HTTP/HTTPS请求。
resp, err := http.Get("http://example.com/")
...
resp, err := http.Post("http://example.com/upload", "image/jpeg", &buf)
...
resp, err := http.PostForm("http://example.com/form",
url.Values{"key": {"Value"}, "id": {"123"}})
程序在使用完response后必须关闭回复的主体。
resp, err := http.Get("http://example.com/")
if err != nil {
// handle error
}
defer resp.Body.Close()
body, err := ioutil.ReadAll(resp.Body)
// ...
GET请求示例
使用net/http
包编写一个简单的发送HTTP请求的Client端,代码如下:
package main
import (
"fmt"
"io/ioutil"
"net/http"
)
func main() {
resp, err := http.Get("https://www.liwenzhou.com/")
if err != nil {
fmt.Printf("get failed, err:%v\n", err)
return
}
defer resp.Body.Close()
body, err := ioutil.ReadAll(resp.Body)
if err != nil {
fmt.Printf("read from resp.Body failed, err:%v\n", err)
return
}
fmt.Print(string(body))
}
将上面的代码保存之后编译成可执行文件,执行之后就能在终端打印liwenzhou.com
网站首页的内容了,我们的浏览器其实就是一个发送和接收HTTP协议数据的客户端,我们平时通过浏览器访问网页其实就是从网站的服务器接收HTTP数据,然后浏览器会按照HTML、CSS等规则将网页渲染展示出来。
带参数的GET请求示例
关于GET请求的参数需要使用Go语言内置的net/url
这个标准库来处理。
func main() {
apiUrl := "http://127.0.0.1:9090/get"
// URL param
data := url.Values{}
data.Set("name", "小王子")
data.Set("age", "18")
u, err := url.ParseRequestURI(apiUrl)
if err != nil {
fmt.Printf("parse url requestUrl failed, err:%v\n", err)
}
u.RawQuery = data.Encode() // URL encode
fmt.Println(u.String())
resp, err := http.Get(u.String())
if err != nil {
fmt.Printf("post failed, err:%v\n", err)
return
}
defer resp.Body.Close()
b, err := ioutil.ReadAll(resp.Body)
if err != nil {
fmt.Printf("get resp failed, err:%v\n", err)
return
}
fmt.Println(string(b))
}
对应的Server端HandlerFunc如下:
func getHandler(w http.ResponseWriter, r *http.Request) {
defer r.Body.Close()
data := r.URL.Query()
fmt.Println(data.Get("name"))
fmt.Println(data.Get("age"))
answer := `{"status": "ok"}`
w.Write([]byte(answer))
}
Post请求示例
上面演示了使用net/http
包发送GET
请求的示例,发送POST
请求的示例代码如下:
package main
import (
"fmt"
"io/ioutil"
"net/http"
"strings"
)
// net/http post demo
func main() {
url := "http://127.0.0.1:9090/post"
// 表单数据
//contentType := "application/x-www-form-urlencoded"
//data := "name=小王子&age=18"
// json
contentType := "application/json"
data := `{"name":"小王子","age":18}`
resp, err := http.Post(url, contentType, strings.NewReader(data))
if err != nil {
fmt.Printf("post failed, err:%v\n", err)
return
}
defer resp.Body.Close()
b, err := ioutil.ReadAll(resp.Body)
if err != nil {
fmt.Printf("get resp failed, err:%v\n", err)
return
}
fmt.Println(string(b))
}
对应的Server端HandlerFunc如下:
func postHandler(w http.ResponseWriter, r *http.Request) {
defer r.Body.Close()
// 1. 请求类型是application/x-www-form-urlencoded时解析form数据
r.ParseForm()
fmt.Println(r.PostForm) // 打印form数据
fmt.Println(r.PostForm.Get("name"), r.PostForm.Get("age"))
// 2. 请求类型是application/json时从r.Body读取数据
b, err := ioutil.ReadAll(r.Body)
if err != nil {
fmt.Printf("read request.Body failed, err:%v\n", err)
return
}
fmt.Println(string(b))
answer := `{"status": "ok"}`
w.Write([]byte(answer))
}
自定义Client
要管理HTTP客户端的头域、重定向策略和其他设置,创建一个Client:
client := &http.Client{
CheckRedirect: redirectPolicyFunc,
}
resp, err := client.Get("http://example.com")
// ...
req, err := http.NewRequest("GET", "http://example.com", nil)
// ...
req.Header.Add("If-None-Match", `W/"wyzzy"`)
resp, err := client.Do(req)
// ...
自定义Transport
要管理代理、TLS配置、keep-alive、压缩和其他设置,创建一个Transport:
tr := &http.Transport{
TLSClientConfig: &tls.Config{RootCAs: pool},
DisableCompression: true,
}
client := &http.Client{Transport: tr}
resp, err := client.Get("https://example.com")
Client和Transport类型都可以安全的被多个goroutine同时使用。出于效率考虑,应该一次建立、尽量重用。
上课时做的代码展示
Server端
:
package main
import (
"fmt"
"io/ioutil"
"net/http"
)
func f2(w http.ResponseWriter, r *http.Request) {
// 对于Get请求,参数都放在url上(Query param),请求体是没有数据的
queryParams := r.URL.Query()
name := queryParams.Get("name")
age := queryParams.Get("age")
fmt.Printf("name : %v , age : %v\n", name, age)
fmt.Println(r.Method)
fmt.Println(ioutil.ReadAll(r.Body)) // 我在服务端打印客户端发来的请求的Body
w.Write([]byte("ok"))
}
func main() {
http.HandleFunc("/Path2/", f2)
http.ListenAndServe("0.0.0.0:9090", nil)
}
Client端`:
Test1
:常规的访问其页面的操作(自定义)
Test2
:经过url.Values
的初始化,在url.Parse
后的urlObj.RawQuery
赋值后构造其的 询问对象,最后利用其询问对象加上对应method
在NewRequest
新的request
,最后在http.DefaultClient.Do()
实现其Client
向Server
端发送数据的效果,经过此过程其实和Test1
实现的效果是一样的,但是其在访问的字段上更有容错性,比如,所访问的字段刚好有?
、=
等特殊字符不能正确解析的情况,可有效规避其问题。
package main
import (
"fmt"
"io/ioutil"
"net/http"
"net/url"
)
func test1() {
resp, err := http.Get("http://127.0.0.1:9090/Path2/?name=hhz&age=18")
if err != nil {
fmt.Printf("Http get failed , err : %v\n", err)
return
}
b, err := ioutil.ReadAll(resp.Body) // 在客户端中接收从服务端发来的Body消息
if err != nil {
fmt.Printf("Read resp.Body failed , err : %v\n", err)
return
}
fmt.Println(string(b))
}
// false : 共用一个clients适用于 请求比较频繁
// true : 定义在局部变量即可 适用于 请求不是特别频繁,用完就关闭
var (
client = http.Client{
Transport: &http.Transport{
DisableKeepAlives: false,
},
}
)
func test2() {
//Parse解析 url 返回其url对象
urlObj, _ := url.Parse("http://127.0.0.1:9090/Path2/")
//构建一个请求对象
data := url.Values{}
data.Set("name", "新垣结衣")
data.Set("age", "18")
//编码 以确保在不同的服务器解析的编码相统一
queryStr := data.Encode()
fmt.Println(queryStr)
//将请求对象的访问字段 进行赋值
urlObj.RawQuery = queryStr
//利用请求对象 加上 方法 返回其新的request(询问)
req, err := http.NewRequest("GET", urlObj.String(), nil)
//自定义其Client发送req, 接收其从服务端返回的 response(响应)
//resp, err := client.Do(req) 使用全局的client
resp, err := http.DefaultClient.Do(req)
b, err := ioutil.ReadAll(resp.Body) // 在客户端中接收从服务端发来的Body消息
if err != nil {
fmt.Printf("Read resp.Body failed , err : %v\n", err)
return
}
defer resp.Body.Close()
fmt.Println(string(b))
}
func main() {
//test1()
test2()
}