【GO】go语言中的HTTP标准库 - http编程

上一节已经学习了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
}

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.hqwc.cn/news/690988.html

如若内容造成侵权/违法违规/事实不符,请联系编程知识网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

Unity开发中导弹路径散射的原理与实现

Unity开发中导弹路径散射的原理与实现 前言逻辑原理代码实现导弹自身脚本外部控制脚本 应用效果结语 前言 前面我们学习了导弹的追踪的效果&#xff0c;但是在动画或游戏中&#xff0c;我们经常可以看到导弹发射后的弹道是不规则的&#xff0c;扭扭曲曲的飞行&#xff0c;然后击…

Reactor Netty 其他-响应式编程-018

&#x1f917; ApiHug {Postman|Swagger|Api...} 快↑ 准√ 省↓ GitHub - apihug/apihug.com: All abou the Apihug apihug.com: 有爱&#xff0c;有温度&#xff0c;有质量&#xff0c;有信任ApiHug - API design Copilot - IntelliJ IDEs Plugin | Marketplace The Nex…

vue3点击添加小狗图片,vue3拆分脚本

我悄悄蒙上你的眼睛 模板和样式 <template><div class"XueXi_Hooks"><img v-for"(dog, index) in dog1List" :src"dog" :key"index" /><button click"addDog1">点我添加狗1</button><hr …

每日一题9:Pandas-填充缺失值

一、每日一题 DataFrame products --------------------- | Column Name | Type | --------------------- | name | object | | quantity | int | | price | int | ---------------------编写一个解决方案&#xff0c;在 quantity 列中将缺失的值 编…

Leetcode 剑指 Offer II 077.排序链表

题目难度: 中等 原题链接 今天继续更新 Leetcode 的剑指 Offer&#xff08;专项突击版&#xff09;系列, 大家在公众号 算法精选 里回复 剑指offer2 就能看到该系列当前连载的所有文章了, 记得关注哦~ 题目描述 给定链表的头结点 head &#xff0c;请将其按 升序 排列并返回 排…

Xilinx 千兆以太网TEMAC IP核 MDIO 配置及物理接口

基于AXI4-Lite接口可以访问MDIO(Management Data Input/Output)接口&#xff0c;而MDIO接口连接MAC外部的PHY芯片&#xff0c;用户可通过AXI4-Lite接口实现对PHY芯片的配置。 1 MDIO接口简介 开放系统互连模型OSI的最低两层分别是数据链路层和物理层&#xff0c;数据链路层的…

音视频入门基础:像素格式专题(2)——不通过第三方库将RGB24格式视频转换为BMP格式图片

音视频入门基础&#xff1a;像素格式专题系列文章&#xff1a; 音视频入门基础&#xff1a;像素格式专题&#xff08;1&#xff09;——RGB简介 音视频入门基础&#xff1a;像素格式专题&#xff08;2&#xff09;——不通过第三方库将RGB24格式视频转换为BMP格式图片 一、引…

xilinx xdma drive 传输8MB以上数据受限的问题

当传输超过8 MB数据时报错error code1359&#xff0c; #define XDMA_MAX_TRANSFER_SIZE (8UL * 1024UL * 1024UL) 可以修改成&#xff1a; #define XDMA_MAX_TRANSFER_SIZE (80UL * 1024UL * 1024UL) VS2019 WDK环境的搭建 先准备好VS WDK的驱动开发环境。需要下载VS、SD…

5 个遥遥领先的大模型 RAG 工具

想象一下拥有一种超能力&#xff0c;让你能够对任何问题或提示生成类似人类的回答&#xff0c;同时还能够利用庞大的外部知识库确保准确性和相关性。这不是科幻小说&#xff0c;这就是检索增强生成&#xff08;RAG&#xff09;的力量。 在本文中&#xff0c;我们将介绍五大遥遥…

NASA数据集——2002-2011年全球18.7 至 89.0 千兆赫的亮度温度、海冰浓度和海冰积雪深度三级网格产品(AE_SI12)数据

AMSR-E/Aqua Daily L3 12.5 km Brightness Temperature, Sea Ice Concentration, & Snow Depth Polar Grids V003 三级网格产品&#xff08;AE_SI12&#xff09;包括 18.7 至 89.0 千兆赫的亮度温度、海冰浓度和海冰积雪深度。 简介 美国国家航空航天局地球观测系统 Aqu…

几个Python处理Excel实际应用

下面介绍四个不同类型的Python处理Excel的经典应用案例&#xff0c;以帮助读者更好地掌握Python处理Excel的技能。 一、读取Excel数据 Python通过pandas库可以轻松地读取Excel数据。pandas库是一个专门用于数据分析和处理的库&#xff0c;它可以将Excel中的数据读取为DataFra…

HIVE调优MapJoin

HIVE调优MapJoin 目录 HIVE调优MapJoin 1.mapjoin &#xff08;1.2以后自动默认启动mapjoin&#xff09; 2.创建表格 3.查询建表 4.通过 explain 展示执行计划 5.Map JOIN 相关设置&#xff1a; 1.mapjoin &#xff08;1.2以后自动默认启动mapjoin&#xff09;…