Mygin 实现简单Http
本篇是完全参考gin的功能,自己手动实现一个类似的功能,帮助自己理解和学习gin框架。
目的
- 简单介绍net/http库以及http.Handler接口
- 实现简单路由请求的功能
标准库启动Web服务
先看用标准库怎样启动一个web服务
import "net/http"
func helloHandler(w http.ResponseWriter, r *http.Request) {
w.Write([]byte("hello world"))
}
func main() {
http.HandleFunc("/hello", helloHandler)
http.ListenAndServe(":8088", nil)
}
设置路由
我设置了一个 /hello 的简单路由,GET请求后返回 hello world。
~ curl 127.0.0.1:8088/hello
hello world
为什么请求路由能够返回hello world,这就要从Golang的http包ListenAndServe.go中查看了。
ListenAndServe源码查看
我已经将源码上的注释翻译成了中文
// ListenAndServe监听TCP网络地址addr,然后调用
//使用处理程序处理传入连接的请求。
//被接受的连接被配置为使能TCP keepalive。
//
//处理程序通常为nil,在这种情况下使用DefaultServeMux。
//
// ListenAndServe总是返回一个非nil错误。
func ListenAndServe(addr string, handler Handler) error {
server := &Server{Addr: addr, Handler: handler}
return server.ListenAndServe()
}
实现Handler接口
- http包中HandleFunc的第二个参数是什么意思?点进官方包去看。
package http
type Handler interface {
ServeHTTP(w ResponseWriter, r *Request)
}
看到第二个参数是要实现 Handler 接口的方法,根据路由匹配交给实现对应的方法处理。
engine.go编写
基于此改造封装mygin/engine.go
package mygin
import (
"net/http"
)
type Engine struct {
}
//ServeHTTP 方法
func (e *Engine) ServeHTTP(w http.ResponseWriter, r *http.Request) {
switch r.URL.Path {
case "/":
w.Write([]byte("success \n"))
case "/hello":
w.Write([]byte("hello world! \n"))
default:
w.Write([]byte("404 NOT FOUND" + r.URL.Path + " \n"))
}
}
func Default() *Engine {
return &Engine{}
}
func (e *Engine) Run(addr string) {
err := http.ListenAndServe(addr, e)
if err != nil {
return
}
}
可以看到我编写了一个engine.go文件,包含了ServeHTTP方法,也就是engine实现了http包中 Handler 接口。http.ListenAndServe第二个参数把engine对象传过去,在http在处理时,就会去调用engine中的ServeHTTP方法。
main.go中调用
package main
import "gophp/mygin"
func main() {
engine := mygin.Default()
engine.Run(":8088")
}
编写好上诉代码后,启动main方法
curl请求
~ curl 127.0.0.1:8088/hello
hello world!
~ curl 127.0.0.1:8088
success
看到上诉结果,代表已经实现了最核心的第一步,可以看出在Golang中,不管什么web框架,所有的请求都是在此基础上运行的。赶快动手去试试吧。