【注意】最后更新于 March 19, 2023,文中内容可能已过时,请谨慎使用。
1. 怎么启动Web服务?
Go
语言标准库内置的net/http
包,可以实现HTTP
服务端。实现HTTP
服务端就是能够启动Web
服务,相当于搭建起了一个Web
服务器。
http.ListenAndServer()
函数用来启动Web
服务,绑定并监听http
端口。
1
2
3
|
func ListenAndServe(addr string, handler Handler)
// addr:监听地址;如 :8080 或者0.0.0.0:8080
// handler:HTTP处理器Handler
|
2.启动Web服务的几种方式
根据不同服务返回的handler
,常见启动Web
服务有以下几种方式。
2.1 http.FileServer: 静态文件服务
http.FileServer()
搭建的服务器只提供静态文件的访问。因为这种web
服务只支持静态文件访问,所以称之为静态文件服务。
1.使用示例
文件目录如下:
1
2
3
4
5
6
7
8
9
10
11
12
|
package main
import (
"net/http"
)
func main() {
runFileServer()
}
// 启动一个文件服务器
func runFileServer() {
// 如果改路径下有index.html,则会优先显示index.html,否则会看到文件目录
http.ListenAndServe(":3000",http.FileServer(http.Dir("./public/")))
}
|
注意: 启动的时候,需求使用go run main.go。否则会报404
2.2 http.HandleFunc: 默认的多路由分发服务
http.HandleFunc()
的作用是注册网络访问的路由。因为它采用的是默认的路由分发任务方式,所以称之为默认的多路由分发服务。
1
2
3
4
5
|
func HandleFunc(pattern string, handler func(ResponseWriter, *Request)) {
DefaultServeMux.HandleFunc(pattern, handler)
}
// pattern:请求路径的匹配模式
// handler:函数类型,表示这个请求需要处理的事情,其实就是Handler接口中的ServeHTTP()方法。
|
ServeHTTP()
方法有两个参数,其中第一个参数是ResponseWriter
类型,包含了服务器端给客户端的响应数据。服务器端往ResponseWriter
写入了什么内容,浏览器的网页源码就是什么内容。第二个参数是一个 *Request
指针,包含了客户端发送给服务器端的请求信息(路径、浏览器类型等)
1.使用示例
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
|
package main
import (
"fmt"
"net/http"
)
func main() {
// 绑定路由/hello
http.HandleFunc("/hello",helloHandle)
// 绑定路由到/test
http.HandleFunc("/test",testHandle)
// 启动服务
err := http.ListenAndServe(":5000",nil)
fmt.Println(err)
}
// 处理路由hello
func helloHandle(w http.ResponseWriter, r *http.Request) {
fmt.Println("访问路由hello")
// 解析url参数
fmt.Println(r.URL.Query())
w.Write([]byte("hello word!"))
}
// 处理路由test
func testHandle(w http.ResponseWriter, r *http.Request) {
fmt.Println("访问路由test")
// 解析url参数,并输出
fmt.Println(r.URL.Query())
w.Write([]byte("test doing!"))
}
// 访问: http://127.0.0.1:5000/test?a=1001
// 访问: http://127.0.0.1:5000/hello?b=990
|
通过http. HandleFunc()
注册网络路由时,http.ListenAndServer()
的第二个参数通常为nil
,这意味着服务端采用默认的http.DefaultServeMux
进行分发处理。
2.3 http.NewServeMux(): 自定义多路由分发服务
http.NewServeMux()
的作用是注册网络访问的多路路由。因为它采用的是自定义的多路由分发任务方式,所以称之为自定义多路由分发服务。
注册网络路由时,如果http.ListenAndServer()
的第二个参数为nil
,那么表示服务端采用默认的http.DefaultServeMux
进行分发处理。也可以自定义ServeMux
。ServeMux
结构体如下
1
2
3
4
5
6
7
8
9
10
11
12
|
// ServeMux结构体源码
type ServeMux struct {
mu sync.RWMutex // 锁,由于请求涉及到并发处理,因此这里有个锁机制
m map[string]muxEntry // 存放具体的路由信息
es []muxEntry // 按照路由长度从大到小的存放处理函数
hosts bool // 标记路由中是否带有主机名
}
// muxEntry是路由的具体条目
type muxEntry struct {
h Handler // 处理函数
pattern string // 路由路径
}
|
3. 自定义多路由实践
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
|
package main
import (
"fmt"
"net/http"
)
// 定义一个接口体,用来实现http.Handler
type MyRoute struct {
}
// 实现http.Handler接口中的ServeHTTP方法
func (m *MyRoute)ServeHTTP(w http.ResponseWriter,r *http.Request) {
path := r.URL.Path
fmt.Println(path)
switch path {
case "/":
w.Write([]byte("首页"))
case "/hello":
w.Write([]byte("say hello"))
case "/test":
w.Write([]byte("test doing"))
default:
http.NotFound(w,r)
}
return
}
func main() {
myRoute := &MyRoute{}
http.ListenAndServe(":10000",myRoute)
}
|
3.1 代码执行流程
使用http.ListenAndServe(":10000",myRoute)
启动服务之后,会发生以下操作
1.实例化http.Server
,并调用ListenAndServe()
1
2
3
4
5
6
|
func ListenAndServe(addr string, handler Handler) error {
// 实例化 Server
server := &Server{Addr: addr, Handler: handler}
// 调用 ListenAndServe()
return server.ListenAndServe()
}
|
2.监听端口
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
|
func (srv *Server) ListenAndServe() error {
if srv.shuttingDown() {
return ErrServerClosed
}
addr := srv.Addr
if addr == "" {
addr = ":http"
}
// 监听端口
ln, err := net.Listen("tcp", addr)
if err != nil {
return err
}
// 启动服务
return srv.Serve(ln)
}
|
3.启动for无限循环,在循环体中Accept
请求,并开启 goroutine
为这个请求服务
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
|
func (srv *Server) Serve(l net.Listener) error {
//...省略N行代码
for {
rw, err := l.Accept()
if err != nil {
//...省略N行代码
}
connCtx := ctx
if cc := srv.ConnContext; cc != nil {
connCtx = cc(connCtx, rw)
if connCtx == nil {
panic("ConnContext returned nil")
}
}
tempDelay = 0
c := srv.newConn(rw)
c.setState(c.rwc, StateNew)
// 开启goroutine为这个请求服务
go c.serve(connCtx)
}
}
|
4.读取每个请求内容,并调用ServeHTTP
1
2
3
4
5
6
7
8
9
10
11
12
|
func (c *conn) serve(ctx context.Context) {
//...省略N行代码
for {
// 读取每个请求内容
w, err := c.readRequest(ctx)
//...省略N行代码
// 调用ServeHTTP
serverHandler{c.server}.ServeHTTP(w, w.req)
//...省略N行代码
}
}
|
5. 判断handler
是否为空,如果为空则把handler
设置成DefaultServeMux
。
1
2
3
4
5
6
7
8
9
10
11
12
|
func (sh serverHandler) ServeHTTP(rw ResponseWriter, req *Request) {
handler := sh.srv.Handler
if handler == nil {
// 如果为空则把handler设置成DefaultServeMux。
handler = DefaultServeMux
}
if req.RequestURI == "*" && req.Method == "OPTIONS" {
handler = globalOptionsHandler{}
}
// 上述示例中,传的是&MyRoute,所以会调用MyRoute.ServeHTTP
handler.ServeHTTP(rw, req)
}
|