上一节已经学习了HTTP的基础知识,本章将学习关于go语言的HTTP编程,最重要的是掌握 net/http 包的用法,以及如何自己编写一个简单的Web服务端,通过客户端访问Server端等。
编写简单的Web 服务器
http.ListenAndServe 启动 Http Server 服务
http.HandleFunc 根据不同的路径将请求路由到不同的处理函数。
路由函数格式固定 ,必须有两个参数 (w http.ResponseWriter,r *http.Request) ,没有返回值
package mainimport ("fmt""net/http"
)func handlerHello(w http.ResponseWriter,r *http.Request) { // 两个参数 ,将返回参数写入到 w, 请求参数在参数r中,这里是简单的例子,所有没有使用到r参数fmt.Fprintf(w,"Hello World!") // 把返回内容写入 http.ResponseWriter
}func handlerBoy(w http.ResponseWriter,r *http.Request) {fmt.Fprintf(w,"hello Boy")
}func handlerGirl(w http.ResponseWriter,r *http.Request) {fmt.Fprintf(w,"hello girl")
}func main() {// 定义路由,将访问不同目录的请求 路由到 不同的处理函数http.HandleFunc("/",handlerHello) // 路由 ,访问 / 根目录是去执行 handlerHello,上面定义好的函数http.HandleFunc("/boy",handlerBoy) // 路由 ,访问/boy目录是去执行 handlerBoyhttp.HandleFunc("/girl",handlerGirl) // 第一个参数是个字符串 ,第二个参数是个函数// 启动HTTP server 服务,ListenAndServe 如果不发生error会一直阻塞。为每一个请求创建一个协程去处理if err := http.ListenAndServe(":8888",nil); err != nil { // 服务端口为 8888fmt.Printf("start http server fail : %s", err)}}
通过浏览器请求server端
运行之后通过浏览器访问 url http://127.0.0.1:8888/,可以看到返回了 Hello World
http://127.0.0.1:8888/boy
通过浏览器访问 url http://127.0.0.1:8888/boy,返回hello Boy
可以看到访问不同的路径返回不同的内容,这就是server端路由的左右。
通过Go编写客户端发起请求
可以都通过简单的http.Get 或者 http.Post 发送请求。
也可以通过较为复杂的 http.NewRequest 发送请求,这种方法更为灵活,可以自定义请求头,Cookie等。
另外请求req ,响应resp 中的内容可以拿出来打印或者做响应处理
package mainimport ("fmt""io""net/http""os""strings""time"
)func main() {get()post()complexHttpRequest()
}// get请求
func get() {resp, err :=http.Get("http://127.0.0.1:8888/boy")if err != nil {panic(err)}defer resp.Body.Close() // 一定要调用 resp.Body.Close() ,否则会协程泄露io.Copy(os.Stdout,resp.Body)// 打印 响应头for k,v := range resp.Header {fmt.Println(k," = ", v)}fmt.Println(resp.Status) // 响应状态fmt.Println(resp.Proto) // http协议
}// post请求
func post() {reader:= strings.NewReader("hello server") // 新建一个io.Reader类型resp , err := http.Post("http://127.0.0.1:8888/girl","text/plain",reader) // 第一个参数是URL,第二个参数是 contentType 类型,第三个参数是请求正文,并不是字符串,而是io.Reader类型if err != nil {panic(err)}io.Copy(os.Stdout,resp.Body)defer resp.Body.Close()// 打印resp.Header 响应头for k,v := range resp.Header {fmt.Println(k, "==>", v)}
}// 复杂的请求
func complexHttpRequest() {reader := strings.NewReader("hello server")// 创建请求,该函数接受三个参数 分别为请求方法,请求的url ,bodyreq , err := http.NewRequest("POST","http://127.0.0.1:8888",reader)if err != nil {panic(err)}// 自定义请求头req.Header.Add("User-Agent","中国")req.Header.Add("MyHeaderKey","MyHeaderValue")// 自定义cookiereq.AddCookie(&http.Cookie{Name:"yhh",Value: "yhh_pwd",Path:"/",Domain: "localhost",Expires: time.Now().Add(time.Duration(time.Hour)),})// 构建clientclient := &http.Client{Timeout: 100 * time.Millisecond, // 设置请求的超时时间, 100毫秒 。}// 提交http请求resp, err := client.Do(req)if err != nil {panic(err)}// 一定要记得关闭defer resp.Body.Close()// 打印resp中的内容io.Copy(os.Stdout,resp.Body)// 打印resp header中的内容for k,v := range resp.Header {fmt.Println(k," = ", v)}}
结构体Request 中文注释
请求中的所有内容基本都在该结构体中,通过学习该结构体加深理解HTTP的基础知识
// Request代表服务器接收到的HTTP请求或客户端要发送的请求。
//
// 字段的语义在客户端和服务器的使用中略有不同。
// 除了下面字段的注释外,还请参阅Request.Write和RoundTripper的文档。
type Request struct {// Method指定HTTP方法(GET、POST、PUT等)。// 对于客户端请求,空字符串表示GET。//// Go的HTTP客户端不支持使用CONNECT方法发送请求。// 有关详情,请参阅Transport的文档。Method string// URL指定正在请求的URI(对于服务器请求)或要访问的URL(对于客户端请求)。//// 对于服务器请求,URL从Request-Line中提供的URI中解析。// 对于大多数请求,除了Path和RawQuery之外的字段将为空。(参见RFC 7230,第5.3节)//// 对于客户端请求,URL的Host指定要连接的服务器,而Request的Host字段可选择地指定要在HTTP请求中发送的Host头的值。URL *url.URL// 传入服务器请求的协议版本。//// 对于客户端请求,这些字段被忽略。HTTP客户端代码始终使用HTTP/1.1或HTTP/2。// 有关详情,请参阅Transport的文档。Proto string // "HTTP/1.0"ProtoMajor int // 1ProtoMinor int // 0// Header包含要发送给服务器的请求头字段,或服务器接收的请求头字段。//// 如果服务器收到带有头行的请求,//// Host: example.com// accept-encoding: gzip, deflate// Accept-Language: en-us// fOO: Bar// foo: two//// 则//// Header = map[string][]string{// "Accept-Encoding": {"gzip, deflate"},// "Accept-Language": {"en-us"},// "Foo": {"Bar", "two"},// }//// 对于传入的请求,Host头将提升为Request.Host字段,并从Header映射中删除。//// HTTP定义了头名称不区分大小写。请求解析器通过使用CanonicalHeaderKey来实现这一点,// 使得首字母和连接符后的任何字符变为大写,其余字符变为小写。//// 对于客户端请求,某些头部,如Content-Length和Connection,在需要时会自动写入,// 并且Header中的值可能会被忽略。请参阅Request.Write方法的文档。Header Header// Body是请求的主体。//// 对于客户端请求,nil主体表示请求没有主体,例如GET请求。// HTTP客户端的Transport负责调用Close方法。//// 对于服务器请求,请求主体始终为非nil,但当没有主体时将立即返回EOF。// 服务器将关闭请求主体。ServeHTTP处理程序不需要这样做。//// Body必须允许在Close的同时调用Read。// 特别是,调用Close应该解除等待输入的Read。Body io.ReadCloser// GetBody定义了一个可选的函数,用于返回Body的新副本。// 当重定向需要多次读取主体时,客户端请求会使用它。// 使用GetBody仍然需要设置Body。//// 对于服务器请求,它未使用。GetBody func() (io.ReadCloser, error)// ContentLength记录相关内容的长度。// 值-1表示长度未知。// 值>= 0表示可以从Body读取给定字节数。//// 对于客户端请求,值为0且Body非nil也被视为未知。ContentLength int64// TransferEncoding列出了从最外层到最内层的传输编码。// 空列表表示“identity”编码。// 当发送和接收请求时,可以通常忽略TransferEncoding;// 在需要时,chunked编码将自动添加和删除。TransferEncoding []string// Close指示在回复此请求后(对于服务器)或发送此请求并读取其响应后(对于客户端)是否关闭连接。//// 对于服务器请求,HTTP服务器会自动处理这一点,并且处理程序不需要此字段。//// 对于客户端请求,设置此字段将防止在相同主机的请求之间重用TCP连接,就像设置了Transport.DisableKeepAlives一样。Close bool// 对于服务器请求,Host指定要搜索URL的主机。// 对于HTTP/1(根据RFC 7230,第5.4节),这要么是“Host”头的值,要么是URL本身中给出的主机名。// 对于HTTP/2,它是“:authority”伪标头字段的值。// 它可以是“host:port”的形式。对于国际域名,Host可能是Punycode或Unicode形式。// 如果需要,可以使用golang.org/x/net/idna将其转换为任何一种格式。// 为了防止DNS重新绑定攻击,服务器处理程序应验证Host头具有处理程序认为自己是权威的值。// ServeMux包含对特定主机名注册的模式,因此可以保护其注册的处理程序。//// 对于客户端请求,Host可选地覆盖要发送的Host头。// 如果为空,则Request.Write方法使用URL.Host的值。Host可能包含国际域名。Host string// Form包含解析的表单数据,包括URL字段的查询参数和PATCH、POST或PUT表单数据。// 只有在调用ParseForm之后才能使用此字段。// HTTP客户端会忽略Form,并使用Body。Form url.Values// PostForm包含来自PATCH、POST或PUT主体参数的解析的表单数据。//// 只有在调用ParseForm之后才能使用此字段。// HTTP客户端会忽略PostForm,并使用Body。PostForm url.Values// MultipartForm是解析的多部分表单,包括文件上传。// 只有在调用ParseMultipartForm之后才能使用此字段。// HTTP客户端会忽略MultipartForm,并使用Body。MultipartForm *multipart.Form// Trailer指定在请求主体之后发送的附加标头。//// 对于服务器请求,Trailer映射最初只包含尾部键,其值为nil。// (客户端声明它将稍后发送哪些尾部。)// 在处理程序从Body中读取时,它不得引用Trailer。// 读取自Body返回EOF后,Trailer可以再次读取,并且如果它们由客户端发送,则将包含非nil值。//// 对于客户端请求,必须将Trailer初始化为包含要稍后发送的尾部键的映射。// 值可以为nil或其最终值。// ContentLength必须为0或-1,以发送分块请求。// 在发送HTTP请求后,可以在读取请求主体的同时更新映射值。// 一旦主体返回EOF,调用者就不能改变Trailer。//// 很少有HTTP客户端、服务器或代理支持HTTP尾部。Trailer Header// RemoteAddr允许HTTP服务器和其他软件记录发送请求的网络地址,通常用于日志记录。// 此字段不会被ReadRequest填充,并且没有定义的格式。// 此包中的HTTP服务器在调用处理程序之前将RemoteAddr设置为“IP:port”地址。// HTTP客户端会忽略此字段。RemoteAddr string// RequestURI是由客户端发送到服务器的Request-Line(RFC 7230,第3.1.1节)的未修改的请求目标。// 通常应使用URL字段。// 在HTTP客户端请求中设置此字段是错误的。RequestURI string// TLS允许HTTP服务器和其他软件记录接收到请求的TLS连接的信息。// 此字段不会由ReadRequest填充。// 此包中的HTTP服务器在调用处理程序之前为启用TLS的连接设置字段;// 否则,它将保留字段为nil。// HTTP客户端会忽略此字段。TLS *tls.ConnectionState// Cancel是一个可选的通道,其关闭指示应将客户端请求视为已取消。// 并非所有的RoundTripper实现都支持Cancel。//// 对于服务器请求,此字段不适用。//// 已弃用:请使用NewRequestWithContext设置Request的上下文,而不是Cancel字段。// 如果一个Request的Cancel字段和上下文都被设置了,那么未定义是否Cancel会被尊重。Cancel <-chan struct{}// Response是导致创建此请求的重定向响应。此字段仅在客户端重定向期间填充。Response *Response// ctx是客户端或服务器上下文。// 应该仅通过复制整个Request使用WithContext来修改它。// 它是未导出的,以防止人们错误地使用Context并改变调用相同请求的调用者持有的上下文。ctx context.Context
}