golang不到20行代码实现路由调度

项目地址

githubjava

本项目依赖

使用标准库实现,无额外依赖mysql

为何须要路由调度层

golang http标准库只能精确匹配请求的URI,而后执行handler。如今通常web项目都至少有个Controller层,以struct实现,根据不一样的请求路径派发到不一样的方法中去。

路由调度器定义

因为golang暂时还不能够动态建立对象(好比java的Class.forName("xxx").newInstance(),xxx是任意存在的class名称)。因此须要手动注册一下controller关系。git

  1. 定义routes保存controller指针
  2. 解析请求过来的URL查询参数,暂定aaction名称,ccontroller名称,本文偷了下懒,没对PATH_INFO作处理,也没有对actionName的首字母自动大写,这个不影响本文要传达的核心内容,有兴趣的读者能够自行实现。
  3. 根据URL中的controllerName找到对应的controller
  4. 使用反射将当前请求对象的*http.Requesthttp.ResponseWriter设置到该Controller
  5. 使用反射以及actionName对应该controller的方法
因为golang的继承不是通常的OOP,因此也没有父子类这种说法,路由注册那里只能使用interface{}

代码实现

app/app.go

该文件为核心调度文件github

package app

import (
    "net/http"
    "reflect"
    "fmt"
)

type application struct {
    routes map[string]interface{}
}

func New() *application {
    return &application{
        routes: make(map[string]interface{}),
    }
}

func (p *application) ServeHTTP(w http.ResponseWriter, r *http.Request) {
    controllerName := r.URL.Query().Get("c")
    actionName := r.URL.Query().Get("a")
    if controllerName == "" || actionName == "" {
        http.Error(w, http.StatusText(http.StatusBadRequest), http.StatusBadRequest)
        return
    }
    route, ok := p.routes[controllerName]
    if !ok {
        http.Error(w, "Controller Not Found", http.StatusNotFound)
        return
    }
    ele := reflect.ValueOf(route).Elem()
    ele.FieldByName("Request").Set(reflect.ValueOf(r))
    ele.FieldByName("Response").Set(reflect.ValueOf(w))
    ele.MethodByName(actionName).Call([]reflect.Value{})
}

func (p *application) printRoutes() {
    for route, controller := range p.routes {
        ele := reflect.ValueOf(controller).Type().String()
        fmt.Printf("%s %s\n", route, ele)
    }
}

func (p *application) Get(route string, controller interface{}) {
    p.routes[route] = controller
}

func (p *application) Run(addr string) error {
    p.printRoutes()
    fmt.Printf("listen on %s\n", addr)
    return http.ListenAndServe(addr, p)
}

app/controller.go

控制器"基类"golang

package app

import "net/http"

type Controller struct {
    Response http.ResponseWriter
    Request  *http.Request
}

controller/site.go

具体业务逻辑类web

package controllers

import (
    "fmt"
    "app"
)

type SiteController struct {
    app.Controller
}

func (p SiteController) Index() {
    fmt.Fprint(p.Response, p.Request.RequestURI)
}

main.go

入口文件sql

package main

import (
    _ "github.com/go-sql-driver/mysql"
    "app"
    "controllers"
)

func main() {
    application := app.New()
    application.Get("site", &controllers.SiteController{})
    application.Run(":8080")
}

运行项目

  1. 启动进程
  2. 访问http://localhost:8080?c=site&a=Index会输出/?c=site&a=Index

写在最后

但愿这个小小的项目能启发到各位读者,早日开发出适合本身的Web框架!app

相关文章
相关标签/搜索