上班无聊用 Go 简易的实现了 trie tree, 然后又基于它实现了一下 HTTP 路由

2016-09-12 16:38:06 +08:00
 vakaka

Trie 树,是一种树形结构,是一种哈希树的变种。典型应用是用于统计,排序和保存大量的字符串(但不仅限于字符串),所以经常被搜索引擎系统用于文本词频统计。它的优点是:利用字符串的公共前缀来减少查询时间,最大限度地减少无谓的字符串比较,查询效率比哈希树高。

type Trie
func New() *Trie
func (trie *Trie) GetNode(v string) (ok bool, node *Node)
func (trie *Trie) Has(pattern string) bool
func (trie *Trie) Match(v string) (bool, *Result)
func (trie *Trie) Put(pattern string, value interface{}) error
func (trie *Trie) SetDelimeter(delimeter string)

基于该 Trie 数实现的 HTTP router

type Context
func NewContent(rw http.ResponseWriter, r *http.Request) *Context
func (context *Context) ParamInt(key string, d ...int) (int, error)
func (context *Context) ParamString(key string, d ...string) string
func (context *Context) WriteString(v string)
type Handler
func NewHanlder() *Handler
func (handler *Handler) Delete(handleFunc func(*Context))
func (handler *Handler) DoDelete(context *Context)
func (handler *Handler) DoGet(context *Context)
func (handler *Handler) DoPatch(context *Context)
func (handler *Handler) DoPost(context *Context)
func (handler *Handler) DoPut(context *Context)
func (handler *Handler) Get(handleFunc func(*Context))
func (handler *Handler) Patch(handleFunc func(*Context))
func (handler *Handler) Post(handleFunc func(*Context))
func (handler *Handler) Put(handleFunc func(*Context))
type HandlerInterface
type Router
func New() *Router
func (router *Router) After(pattern string, midwares ...func(context *Context))
func (router *Router) Before(pattern string, midwares ...func(*Context))
func (router *Router) Delete(pattern string, handlefunc func(*Context))
func (router *Router) Get(pattern string, handlefunc func(*Context))
func (router *Router) Patch(pattern string, handlefunc func(*Context))
func (router *Router) Post(pattern string, handlefunc func(*Context))
func (router *Router) Put(pattern string, handlefunc func(*Context))
func (router *Router) Router(pattern string, handler HandlerInterface)
func (router *Router) ServeHTTP(rw http.ResponseWriter, r *http.Request)

示例

package main

import (
    "fmt"
    "net/http"
    "os"

    "github.com/importcjj/trie.go/router"
)

func Helloworld(ctx *router.Context) {
    ctx.WriteString("hello, world!")
}

func ParamHandler(ctx *router.Context) {
    username := ctx.ParamString("username")
    text := fmt.Sprintf("hi, %s", username)
    ctx.WriteString(text)
}

var PageResource = &router.Handler{
    OnGet: func(ctx *router.Context) {
        filepath := ctx.ParamString("filepath")
        text := fmt.Sprintf("Get page %s", filepath)
        ctx.WriteString(text)
    },

    OnPost: func(ctx *router.Context) {
        filepath := ctx.ParamString("filepath")
        text := fmt.Sprintf("Post page %s", filepath)
        ctx.WriteString(text)
    },
}

// BasicAuth is a Midwares
func BasicAuth(ctx *router.Context) {
    fmt.Fprintln(os.Stderr, ctx.Request.URL, "Call Basic Auth.")
}

// BeforeMetric mark a time point when the request start.
func BeforeMetric(ctx *router.Context) {
    // just a example, so use the params map to
    // record the time.
    ctx.Params["time"] = time.Now().Format("Mon Jan 2 15:04:05 -0700 MST 2006")
}

// AfterMetric log the time spent to handle the requeset.
func AfterMetric(ctx *router.Context) {
    start, _ := time.Parse("Mon Jan 2 15:04:05 -0700 MST 2006", ctx.Params["time"])
    dur := time.Since(start)
    fmt.Fprintf(os.Stderr, "%s spent", dur.String())
}

var r = router.New()

func init() {
    r.Get("/hello/world", Helloworld)
    r.Get("/hi/<username:str>", ParamHandler)
    // restful api style, this pattern can match such as
    // "/page/hi.html" "/page/static/inde.html" eta.
    r.Router("/page/<filepath:*>", PageResource)

    r.Before("/", BasicAuth, BeforeMetric)
    r.After("/", AfterMetric)
}

func main() {
    server := &http.Server{
        Addr:    ":8080",
        Handler: r,
    }
    server.ListenAndServe()
}

项目地址: https://github.com/importcjj/trie.go

没有花很多时间,没有进行彻底测试。不过简单测试了一下,没有发现什么问题,性能也还过的去。如果可以的话,希望大家去试试,提一点意见。如果觉得写得可以的话,欢迎 star 和 fork 哦!然后,基于该 HTTP router 可以实现一个简单的 web 框架(懒得搞了)。有兴趣的同学可以试试。

1619 次点击
所在节点    Go 编程语言
3 条回复
JamesRuan
2016-09-12 20:47:47 +08:00
上班时写得东西适合开源吗?

我这个库虽然是为公司使用写的,但是却是 100%的下班时间+个人设备的产物。

https://github.com/jamesruan/sodium
vakaka
2016-09-12 23:17:48 +08:00
@JamesRuan 不是为公司写的,主要是没事干。
zijing07
2016-11-16 09:52:28 +08:00
公司不管你做别的事情么

这是一个专为移动设备优化的页面(即为了让你能够在 Google 搜索结果里秒开这个页面),如果你希望参与 V2EX 社区的讨论,你可以继续到 V2EX 上打开本讨论主题的完整版本。

https://www.v2ex.com/t/305714

V2EX 是创意工作者们的社区,是一个分享自己正在做的有趣事物、交流想法,可以遇见新朋友甚至新机会的地方。

V2EX is a community of developers, designers and creative people.

© 2021 V2EX