Go For Web:Golang http 包详解(源码剖析)

这篇具有很好参考价值的文章主要介绍了Go For Web:Golang http 包详解(源码剖析)。希望对大家有所帮助。如果存在错误或未考虑完全的地方,请大家不吝赐教,您也可以点击"举报违法"按钮提交疑问。

前言:

本文作为解决如何通过 Golang 来编写 Web 应用这个问题的前瞻,对 Golang 中的 Web 基础部分进行一个简单的介绍。目前 Go 拥有成熟的 Http 处理包,所以我们去编写一个做任何事情的动态 Web 程序应该是很轻松的,接下来我们就去学习了解一些关于 Web 的相关基础,了解一些概念,以及 Golang 是如何运行一个 Web 程序的。
文章预计分为四个部分逐步更新
2023-04-13 星期四 一更 全文共计约 3800 字 阅读大约花费 5 分钟
2023-04-14 星期五 二更(两篇) 全文共计约 2000 字 阅读大概花费 4 分钟
2023-04-14 星期五 三更 全文共计约 2000 字 阅读大概花费 5 分钟


文章目录:

  1. Web 的工作方式
  2. 用 Go 搭建一个最简单的 Web 服务
  3. 了解 Golang 运行 web 的原理
  4. Golang http 包详解(源码剖析)

正文:

Golang http 包详解(源码剖析)

前面小节我们认识了 Web 的工作方式,也成功用 Go 搭建了一个最简单的 Web 服务了解了 Golang 运行 Web 的原理。现在我们详细地去解剖以下 http 包,看看它如何实现整个过程的

Go 的 http 包中有两个核心功能:Conn 、ServeMux

Conn 的 goroutine

与我们使用其他语言编写 http 服务器不同, Go为了实现高并发和高性能,使用了 goroutines 来处理 Conn 的读写事件。这样让每个请求都能保持独立,相互不会阻塞,可以高效地响应网络事件,这是 Go 高效的保证。

根据上一节,我们知道 Go 在等待客户端请求里面是这样写的:

点击查看代码
c, err := srv.newConn(rw)
if ree != nil {
	continue
}
go c.serve()

这段代码中,客户端的每一次请求都会创建一个 Conn,这个 Conn 里面保存了这次请求的信息,然后再传递到对应的 handler,该handler中便可以读取到相应的 header 信息,这样保证了每个请求的独立性。

ServeMux 的自定义

在之前我们 使用 conn.server 的时候,其实内部是调用了 http 包默认的路由器也就是DefaultServeMux,通过这个路由器把本次请求的信息传递到了后端的处理函数。那么这个路由器是怎么实现的呢?

结构如下:

  • 首先是一个 自定义类型结构体 ServeMux 其中包含一个
    和一个 路由规则
    Go For Web:Golang http 包详解(源码剖析)

  • 路由规则中一个 string 对应一个 mux 实体,我们来看看 muxEntry 它也是一个自定义类型结构体,包含一个 布尔值,一个Handler 处理函数
    Go For Web:Golang http 包详解(源码剖析)

  • 最后再来看看 Handler 的定义,它其实是一个接口,实现了 ServeHTTP 这个函数
    Go For Web:Golang http 包详解(源码剖析)

这个时候我们可以回过头来看我们之前自己写的 Web 服务器

点击查看代码
// Handler处理函数
func sayhelloName(w http.ResponseWriter, r *http.Request) {
	r.ParseForm() // 解析参数,默认不会解析
	fmt.Println(r.Form)// 以下这些信息是输出到服务端的打印信息:请求表单form、路径path、格式scheme
	fmt.Println("path", r.URL.Path)
	fmt.Println("scheme", r.URL.Scheme)
	fmt.Println(r.Form["url_long"])
	for k, v := range r.Form {
		fmt.Println("key:", k)
		fmt.Println("val:", strings.Join(v, ""))
	}
	fmt.Fprintln(w, "Hello astaxie!") // 输出到客户端
}
调用: `http.HandleFunc("/", sayhelloName) // 设置访问的路由`

我们会发现,我们自己写的 sayhelloName 函数并没有实现 ServeHTTP 这个函数,也就是说按照常理我们并没有实现 Handler 这个接口,那我们是怎么添加的?

原来, http 包里面还定义了一个自定义函数类型 HandlerFunc,而我们定义的函数 sayhelloName 就是这个 HandlerFunc 调用之后的结果,这个自定义函数类型默认会实现 ServeHTTP 这个方法,即我们调用了 HandlerFunc(f)强制类型转换 f 成为了 HandlerFunc 类型,这样 f 就拥有了ServeHTTP 方法
Go For Web:Golang http 包详解(源码剖析)

路由器里存储好了相应的路由规则(Response / Request)之后,那么具体的请求又是怎么分发的呢?
路由器接收到请求之后调用 mux.handler(r).ServeHTTP(w,r)
也就是调用对应路由的 handler 的 ServerHTTP 接口,让我们来看看
mux.handler(r)是怎么处理的↓
Go For Web:Golang http 包详解(源码剖析)

我们可以看到它是根据用户请求的 URL 和路由器里面存储的 map 去匹配的,当匹配到之后返回存储的 handler,调用这个 handler 的 ServeHTTP 接口就可以执行相应的函数了

通过上面的介绍,我们大致了解了整个构建路由的过程,Go其实支持外部实现的路由器 而 ListenAndServe 的第二个参数就是用来配置外部路由器的,它是一个 Handler 接口,所以我们的外部路由只要实现了 Handler 接口就可以发挥作用,因此我们可以在自己实现的路由器的 ServeHTTP 里面实现自定义的路由功能

贴个代码↓

点击查看代码
package main

import (
	"fmt"
	"net/http"
)

type MyMux struct {
}

func (p *MyMux) ServeHTTP(w http.ResponseWriter, r *http.Request) {
	if r.URL.Path == "/" {
		sayhelloName2(w, r)
		return
	} else {
		http.NotFound(w, r)
		return
	}
}

func sayhelloName2(w http.ResponseWriter, r *http.Request) {
	fmt.Fprintf(w, "Hello myroute!")
}

func main() {
	mux := &MyMux{}
	http.ListenAndServe(":9090", mux)
}

实现效果:
Go For Web:Golang http 包详解(源码剖析)

Go 代码的执行流程

最后我们来梳理一下整个代码的执行过程

  • 首先调用 Http.HandleFunc
    按照顺序做了这几件事:
  1. 调用了 DefaultServeMux 的 HandleFunc
  2. 调用了 DefaultServeMux 的 Handler
  3. 往 DefaultServeMux 的 map[string]muxEntry 中增加对应的handler 和 路由规则
  • 其次调用 http.ListenAndServe(":9090",nil)
    按顺序做了这几件事:
  1. 实例化 Server

  2. 调用 Server 的 ListenAndServer()

  3. 调用 net.Listen("tcp", addr)监听端口

  4. 启动一个 for 循环,在循环题中 Accept 请求

  5. 对每一个请求实例化一个 Conn,并且开启一个 goroutine 为这个请求开一个 go.c.serve()

  6. 读取每个请求的内容 w, err := c.readRequest()

  7. 判断 handler 是否为空,如果没有就设置 handler(默认设置)

  8. 调用 handler 的ServeHTTP

  9. 进入到 DefaultServerMux.ServeHTTP

  10. 根据 request 选择 handler, 并且进去到这个 handler 的 ServerHTTP
    Go For Web:Golang http 包详解(源码剖析)

  11. 选择 handler
    A 判断是否有路由能满足这个 request (循环遍历 ServerMux 的 muxEntry)
    B 如果满足,则调用这个路由 handler 的 ServeHTTP
    C 如果不满足,则调用 NotFoundHandler 的 ServeHTTP

总结

到这里为止我们从第一章介绍了 HTTP 协议,DNS 解析过程,了解了 Web 的工作方式,第二章分别用 Go 搭建一个最简单的 Web 服务,并且了解 Golang 运行 web 的原理,在最后一章,我们还深入到 net/http 包中的源码里为大家揭开了更底层的原理

既然对 Go 开发 Web 有了初步的了解,接下来我们就可以有十足的信心去学习更多 Go For Web 的后续内容了!

关于 Golang 基础部分 以及 计算机网络部分读者可以参阅我的往期 blog👇
Goalng:基础复习一遍过

漫谈计算机网络:网络层 ------ 重点:IP协议与互联网路由选择协议

以上

看完记得留下一个👍文章来源地址https://www.toymoban.com/news/detail-413490.html

到了这里,关于Go For Web:Golang http 包详解(源码剖析)的文章就介绍完了。如果您还想了解更多内容,请在右上角搜索TOY模板网以前的文章或继续浏览下面的相关文章,希望大家以后多多支持TOY模板网!

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处: 如若内容造成侵权/违法违规/事实不符,请点击违法举报进行投诉反馈,一经查实,立即删除!

领支付宝红包 赞助服务器费用

相关文章

  • 【Golang】Golang进阶系列教程--为什么 Go for-range 的 value 值地址每次都一样?

    循环语句是一种常用的控制结构,在 Go 语言中,除了 for 以外,还有一个 range ,可以使用 for-range 循环迭代数组、切片、字符串、map 和 channel 这些数据类型。 但是在使用 for-range 循环迭代数组和切片的时候,是很容易出错的,甚至很多老司机一不小心都会在这里

    2024年02月15日
    浏览(51)
  • GO 中高效 int 转换 string 的方法与高性能源码剖析

    Go 语言 中,将整数(int)转换为字符串(string)是一项常见的操作。 本文将从逐步介绍几种在 Go 中将 int 转换为 string 的常见方法,并重点剖析这几种方法在性能上的特点。另外,还会重点介绍 FormatInt 高效的算法实现。 最直接且常用的方法是使用 strconv 包中的 Itoa 函数。

    2024年01月21日
    浏览(36)
  • 【golang】http.ListenAndServe源码解析

    ListenAndServe监听TCP地址addr,并且会使用handler参数调用Serve函数处理接收到的连接。handler参数一般会设为nil,此时会使用DefaultServeMux。 接下来我们看一下这个函数的主要源码流程。 Server 定义运行HTTP服务器的参数。Server的零值是一个有效的配置。 第一层相当于封装了一下创建

    2024年04月08日
    浏览(49)
  • golang http transport源码分析

    Golang http库在日常开发中使用会很多。这里通过一个demo例子出发,从源码角度梳理golang http库底层的数据结构以及大致的调用流程 先看看http.Client结构体,如下: Transport:http client实际发送请求结构体 Timeout:http client请求超时设置(后续在详细分析) 在看Transport结构体(net/http

    2024年02月11日
    浏览(29)
  • golang通过go-git下载gitlab源码

    1 申请令牌 方法1:具体项目下申请: 方法2:全局申请 2 获取token 3 下载代码 替换下面: username token 参考: https://docs.gitlab.cn/jh/user/profile/personal_access_tokens.html

    2024年01月24日
    浏览(40)
  • Golang 包详解以及go mod

    包(package)是多个 Go 源码的集合,是一种高级的代码复用方案,Go 语言为我们提供了 很多内置包,如 fmt、strconv、strings、sort、errors、time、encoding/json、os、io 等。 Golang 中的包可以分为三种 :1、系统内置包 2、自定义包 3、第三方包 系统内置包: Golang 语言给我们提供的内置

    2024年02月14日
    浏览(41)
  • Go 工具链详解(四): Golang环境变量设置和查看工具 go env

    go env 是 Go 工具链中的一个命令,用于设置和查看当前 Golang 环境的相关信息,对于理解、编译和运行 Golang 程序非常有用。 go 提供的命令及 go 程序的编译运行都会使用到环境变量,如果未设置对应的环境变量,go 则会使用其默认设置。默认情况下,env 以 shell 脚本(在Windo

    2024年02月16日
    浏览(37)
  • Go For Web:踏入Web大门的第一步——Web 的工作方式

    本文作为解决如何通过 Golang 来编写 Web 应用这个问题的前瞻,对 Golang 中的 Web 基础部分进行一个简单的介绍。目前 Go 拥有成熟的 Http 处理包,所以我们去编写一个做任何事情的动态 Web 程序应该是很轻松的,接下来我们就去学习了解一些关于 Web 的相关基础,了解一些概念,

    2023年04月13日
    浏览(31)
  • 【Golang | Gin】net/http和Gin起web服务时的简单对比

    Gin 是一个用 Go (Golang) 编写的 Web 框架,详细介绍参考官网:https://gin-gonic.com/zh-cn/docs/introduction/ 开始学习Gin之前,我们先首先回顾下使用net/http起一个简单的helloworld服务 注: 1、 http.HandleFunc : 使用一个默认的 DefaultServeMux 来注册路由信息。 / 是一个 pattern , greet 是一个 ha

    2024年02月08日
    浏览(39)
  • 【Golang】VsCode下开发Go语言的环境配置(超详细图文详解)

    📓推荐网站(不断完善中):个人博客 📌个人主页:个人主页 👉相关专栏:CSDN专栏、个人专栏 🏝立志赚钱,干活想躺,瞎分享的摸鱼工程师一枚 ​ 话说在前,Go语言的编码方式是 UTF-8 ,理论上你直接使用文本进行编辑也是可以的,当然为了提升我们的开发效率我们还是需

    2024年02月07日
    浏览(75)

觉得文章有用就打赏一下文章作者

支付宝扫一扫打赏

博客赞助

微信扫一扫打赏

请作者喝杯咖啡吧~博客赞助

支付宝扫一扫领取红包,优惠每天领

二维码1

领取红包

二维码2

领红包