zoukankan      html  css  js  c++  java
  • go语言实战向导

    版权声明:本文由魏佳原创文章,转载请注明出处: 
    文章原文链接:https://www.qcloud.com/community/article/173

    来源:腾云阁 https://www.qcloud.com/community

    使用go语言做后台服务已经有3年了,通过项目去检验一个又一个的想法,然后不断总结,优化,最终形成了自己的一整套体系,小到一个打印对象的方法,大到一个web后台项目最佳实践指导,这一点一滴都是在不断的实践中进化开来。以下内容将是一次整体的汇报,各位看官如有兴致,请移步GitHub 关注最新的代码变更。

    wsp (go http webserver)

    实现初衷

    • 简单可依赖,充分利用go已有的东西,不另外增加复杂、难以理解的东西,这样做的好处包括:更容易跟随go的升级而升级,降低使用者学习成本
    • yii提供的controller/action的路由方式比较常用,在wsp里实现一套
    • java annotation的功能挺方便,在wsp里,通过注释来实现过滤器方法的调用定义
    • 不能因为wsp的引入而降低原生go http webserver的性能

    使用场景

    • 以http webserver方式对外提供服务
    • 后台接口服务

    使用案例

    大型互联网社交业务

    实现方式

    路由自动生成,按要求提供controller/action的实现代码,wsp执行后会分析项目代码,自动生成路由表并记录在文件demo/WSP.go里,controller/action定义代码必须符合函数定义:func(http.ResponseWriter, *http.Request),并且是带receiver的methoddemo_set.go

    package controller
    
    import (
        "net/http"
    
        "github.com/simplejia/wsp/demo/service"
    )
    
    // @prefilter("Login", {"Method":{"type":"get"}})
    // @postfilter("Boss")
    func (demo *Demo) Set(w http.ResponseWriter, r *http.Request) {
        key := r.FormValue("key")
        value := r.FormValue("value")
        demoService := service.NewDemo()
        demoService.Set(key, value)
    
        json.NewEncoder(w).Encode(map[string]interface{}{
            "code": 0,
        })
    }
    

    WSP.go

    // generated by wsp, DO NOT EDIT.
    
    package main
    
    import "net/http"
    import "time"
    import "github.com/simplejia/wsp/demo/controller"
    import "github.com/simplejia/wsp/demo/filter"
    
    func init() {
        http.HandleFunc("/Demo/Get", func(w http.ResponseWriter, r *http.Request) {
            t := time.Now()
            _ = t
            var e interface{}
            c := new(controller.Demo)
            defer func() {
                e = recover()
                if ok := filter.Boss(w, r, map[string]interface{}{"__T__": t, "__C__": c, "__E__": e}); !ok {
                    return
                }
            }()
            c.Get(w, r)
        })
    
        http.HandleFunc("/Demo/Set", func(w http.ResponseWriter, r *http.Request) {
            t := time.Now()
            _ = t
            var e interface{}
            c := new(controller.Demo)
            defer func() {
                e = recover()
                if ok := filter.Boss(w, r, map[string]interface{}{"__T__": t, "__C__": c, "__E__": e}); !ok {
                    return
                }
            }()
            if ok := filter.Login(w, r, map[string]interface{}{"__T__": t, "__C__": c, "__E__": e}); !ok {
                return
            }
            if ok := filter.Method(w, r, map[string]interface{}{"type": "get", "__T__": t, "__C__": c, "__E__": e}); !ok {
                return
            }
            c.Set(w, r)
        })
    
    }
    
    • wsp分析项目代码,寻找符合要求的注释(见demo/controller/demo_set.go),自动生成过滤器调用代码在文件demo/WSP.go里,filter注解分为前置过滤器(prefilter)和后置过滤器(postfilter),格式如:@prefilter({json body}),{json body}代表传入参数,符合json array定义格式(去掉前后的中括号),可以包含string值或者object值,filter函数定义满足:func (http.ResponseWriter*http.Requestmap[string]interface{}) bool,过滤器函数如下: method.go
    package filter
    import (
        "net/http"
        "strings"
    )
    
    func Method(w http.ResponseWriter, r *http.Request, p map[string]interface{}) bool {
        method, ok := p["type"].(string)
        if ok && strings.ToLower(r.Method) != strings.ToLower(method) {
            http.Error(w, "405 Method Not Allowed", http.StatusMethodNotAllowed)
            return false
        }
        return true
    }
    

    filter输入参数map[string]interface{},会自动设置"T",time.Time类型,值为执行起始时间,可用于耗时统计,"C",{Controller}类型,值为{Controller}实例,可通过接口方式存取相关数据(这种方式存取数据较context方式更简单实用),"E",值为recover()返回值,用于检测错误并处理(后置过滤器必须recover())

    • 项目main.go代码示例 main.go
    package main
    
    import (
        "log"
    
        "github.com/simplejia/clog"
        "github.com/simplejia/lc"
    
        "net/http"
    
        _ "github.com/simplejia/wsp/demo/clog"
        _ "github.com/simplejia/wsp/demo/conf"
        _ "github.com/simplejia/wsp/demo/mysql"
        _ "github.com/simplejia/wsp/demo/redis"
    )
    
    func init() {
        lc.Init(1e5)
    
        http.HandleFunc("/", func(w http.ResponseWriter, r *http.Request) {
            http.NotFound(w, r)
        })
    }
    
    func main() {
        clog.Info("main()")
    
        log.Panic(http.ListenAndServe(":8080", nil))
    }
    

    miscellaneous

    • 通过wrk压测工具在同样环境下(8核,8g),wsp空跑qps:9万,beego1.7.1空跑qps:5.5万
    • 更方便加入middleware(func(http.Handler) http.Handler),其实更推荐通过定义过滤器的方式支持类似功能
    • 更方便编写如下的测试用例:
      test (测试用例运行时需要用到项目配置文件,所以请在test目录生成../clog,../conf,../mysql,../redis的软链接)

    demo

    提供一个简单易扩展的项目stub

    实现初衷

    • 简单可依赖,充分利用go已有的东西,不另外增加复杂、难以理解的东西,这样做的好处包括:更容易跟随go的升级而升级,降低使用者学习成本
    • 提供常用组件的简单包装,如下:
      • config,提供项目主配置文件自动解析,见conf
      • redis,使用(github.com/garyburd/redigo),提供配置文件自动解析,见redis
      • mysql,使用(database/sql),提供配置文件自动解析,见mysql,同时为了方便对象映射,提供了最常用的orm组件供选择使用,见orm

    项目编写指导意见

    • 目录结构:
    ├── WSP.go
    ├── clog
    │   └── clog.go
    ├── conf
    │   ├── conf.go
    │   └── conf.json
    ├── controller
    │   ├── base.go
    │   ├── demo.go
    │   ├── demo_get.go
    │   └── demo_set.go
    ├── demo
    ├── filter
    │   ├── boss.go
    │   ├── login.go
    │   └── method.go
    ├── main.go
    ├── model
    │   ├── demo.go
    │   ├── demo_get.go
    │   └── demo_set.go
    ├── mysql
    │   ├── demo_db.json
    │   └── mysql.go
    ├── redis
    │   ├── demo.json
    │   └── redis.go
    ├── service
    │   ├── demo.go
    │   ├── demo_get.go
    │   └── demo_set.go
    └── test
        ├── clog -> ../clog
        ├── conf -> ../conf
        ├── demo_get_test.go
        ├── demo_set_test.go
        ├── init_test.go
        ├── mysql -> ../mysql
        └── redis -> ../redis
    
    • controller目录:负责request参数解析,service调用
    • service目录:负责逻辑处理,model调用
    • model目录:负责数据处理

    接口实现上,建议一个接口对应一个文件,如controller/demo_get.go, service/demo_get.go, model/demo_get.go

    lc (local cache)

    实现初衷

    • 纯用redis做缓存,相比lc,redis有网络调用开销,反复调用多次,延时急剧增大,当网络偶尔出现故障时,我们的数据接口也就拿不到数据,但lc里的数据就算是超过了设置的过期时间,我们一样能拿到过期的数据做备用
    • 使用mysql,当缓存失效,有数据穿透的风险,lc自带并发控制,有且只允许同一时间同一个key的唯一一个client穿透到数据库,其它直接返回lc缓存数据

    特性

    • 本地缓存
    • 支持Get,Set,Mget,Delete操作
    • 当缓存失效时,返回失效标志同时,还返回旧的数据,如:v, ok := lc.Get(key),当key已经过了失效时间了,并且key还没有被lru淘汰掉,v是之前存的值,ok返回false
    • 实现代码没有用到锁
    • 使用到lru,淘汰长期不用的key
    • 结合lm使用更简单快捷

    demo

    lc_test.go

    package lc
    
    import (
        "testing"
        "time"
    )
    
    func init() {
        Init(65536) // 使用lc之前必须要初始化
    }
    
    func TestGetValid(t *testing.T) {
        key := "k"
        value := "v"
        Set(key, value, time.Second)
        time.Sleep(time.Millisecond * 10) // 给异步处理留点时间
        v, ok := Get(key)
        if !ok || v != value {
            t.Fatal("")
        }
    }
    

    lm (lc+redis+[mysql|http] glue)

    实现初衷

    写redis+mysql代码时(还可能加上lc),示意代码如下:

    func orig(key string) (value string) {
        value = redis.Get(key)
        if value != "" {
            return
        }
        value = mysql.Get(key)
        redis.Set(key, value)
        return
    }
    // 如果再加上lc的话
    func orig(key string) (value string) {
        value = lc.Get(key)
        if value != "" {
            return
        }
        value = redis.Get(key)
        if value != "" {
            lc.Set(key, value)
            return
        }
        value = mysql.Get(key)
        redis.Set(key, value)
        lc.Set(key, value)
        return
    }
    

    有了lm,再写上面的代码时,一切变的那么简单 lm_test.go

    func tGlue(key, value string) (err error) {
        err = Glue(
            key,
            &value,
            func(p, r interface{}) error {
                _r := r.(*string)
                *_r = "test value"
                return nil
            },
            func(p interface{}) string {
                return fmt.Sprintf("tGlue:%v", p)
            },
            &LcStru{
                Expire: time.Millisecond * 500,
                Safety: false,
            },
            &McStru{
                Expire: time.Minute,
                Pool: pool,
            },
        )
        if err != nil {
            return
        }
        return
    }
    

    功能

    自动添加缓存代码,支持lc, redis,减轻你的心智负担,让你的代码更加简单可靠,少了大段的冗余代码,复杂的事全交给lm自动帮你做了
    支持Glue[Lc|Mc]及相应批量操作Glues[Lc|Mc],详见lm_test.go示例代码

    注意

    lm.LcStru.Safety,当置为true时,对lc在并发状态下返回的nil值不接受,因为lc.Get在并发状态下,同一个key返回的value有可能是nil,并且ok状态为true,Safety置为true后,对以上情况不接受,会继续调用下一层逻辑

    orm (配合sql.Rows使用的超简单数据到对象映射功能函数)

    实现初衷

    • database/sql包,Db.Query返回的sql.Rows,通过Rows.Scan方式示例代码如下:
    rows, err := db.Query("SELECT ...")
    defer rows.Close()
    for rows.Next() {
        var id int
        var name string
        err = rows.Scan(&id, &name)
    }
    err = rows.Err()
    ...
    

    但实际项目场景里,我们更想这样:

    rows, err := db.Query("SELECT ...")
    defer rows.Close()
    var d []*stru
    err = Rows2Strus(rows, &d)
    

    这就是一种简单的对象映射,通过转为对象的方式,我们的代码更方便处理了

    功能

    一共提供四种场景的使用方法:

    • Rows2Strus, sql.Rows转为struct slice

    • sql.Rows转为struct,等同db.QueryRow

    • Rows2Cnts, sql.Rows转为int slice

    • Rows2Cnt, sql.Rows转为int,用于select count(1)操作

    支持tag: orm,如下:

    type Demo struct {
        Id int
        DemoName string `orm:"demo_name"` // 映射成demo_name字段
    }
    

    支持匿名成员,如下:

    type C struct {
        Id int
    }
    type P struct {
        C  // 映射成id字段
        Name string
    }
    

    支持snakecase配置,通过设置orm.IsSnakeCase = true,如下:

    type Demo struct {
        Id int
        DemoName string // 映射成demo_name字段
    }
    

    demo

    orm_test.go

    cmonitor

    功能

    用于进程监控,管理

    实现

    • 被监控进程启动后,按每300ms执行一次状态检测(通过发signal0信号检测),每个被监控进程在一个独立的协程里被监测。
    • cmonitor启动后会监听一个http端口用于接收管理命令(start|stop|status|...)

    使用方法

    配置文件:conf.json (json格式,支持注释) conf.json

    {
        "env": "dev", // 配置运行环境
        "envs": {
            "dev": {
                "port": 29118, // 配置监听端口
                "rootpath": "/home/simplejia/tools/go/ws/src/github.com/simplejia",
                "environ": "ulimit -n 65536", // 配置环境变量
                "svrs": {
                    // demo
                    "demo": "wsp/demo/demo" // key: 名字 value: 将与rootpath拼接在一起运行
                },
                "log": {
                    "mode": 3, // 0: none, 1: localfile, 2: collector (数字代表bit位)
                    "level": 15 // 0: none, 1: debug, 2: warn 4: error 8: info (数字代表bit位)
                }
            },
            "test": {
                "port": 29118, // 配置监听端口
                "rootpath": "/home/simplejia/tools/go/ws/src/github.com/simplejia",
                "environ": "ulimit -n 65536", // 配置环境变量
                "svrs": {
                    // demo 
                    "demo": "wsp/demo/demo"
                },
                "log": {
                    "mode": 3, // 0: none, 1: localfile, 2: collector (数字代表bit位)
                    "level": 15 // 0: none, 1: debug, 2: warn 4: error 8: info (数字代表bit位)
                }
            },
            "prod": {
                "port": 29118, // 配置监听端口
                "rootpath": "/home/simplejia/tools/go/ws/src/github.com/simplejia",
                "environ": "ulimit -n 65536", // 配置环境变量
                "svrs": {
                    // demo 
                    "demo": "wsp/demo/demo"
                },
                "log": {
                    "mode": 2, // 0: none, 1: localfile, 2: collector (数字代表bit位)
                    "level": 14 // 0: none, 1: debug, 2: warn 4: error 8: info (数字代表bit位)
                }
            }
        }
    }
    
    • 运行方法:cmonitor.sh [start|stop|restart|status|check]
    • 进程管理:cmonitor -[h|status|start|stop|restart] [all|["svrname"]]

    注意

    • cmonitor的运行日志通过clog上报,也可记录在本地cmonitor.log日志文件里,注意:此cmonitor.log日志文件不会被切分,所以尽量保持较少的日志输出,建议通过clog方式上报日志
    • cmonitor启动监控进程后,被监控进程控制台日志cmonitor.log会输出到相应进程目录,最多保存30天,历史日志以cmonitor.{day}.log方式备份
    • 当cmonitor启动时,会根据conf.json配置启动所有被监控进程,当被监控进程已经启动过,并且符合配置要求时,cmonitor会自动将其加入监控列表
    • cmonitor会定期检查进程运行状态,如果进程异常退出,cmonitor会反复重试拉起,并且记录日志
    • 当被监控进程为多进程运行模式,cmonitor只监控管理父进程(子进程应实现检测父进程运行状态,并随父进程退出而退出)
    • 被监控进程以nohup方式启动,所以你的程序就不要自己设定daemon运行了
    • 每分钟通过ps方式检测一次进程状态,如果出现任何异常,比如有多份进程启动等,记日志
    • 由于cmonitor会同时启动内部httpserver(绑内网ip),所以也支持远程管理,比如在浏览器里输入:http://xxx.xxx.xxx.xxx:29118/?command=status&service=all

    demo

    $ cmonitor -status all
    
    *****STATUS OK SERVICE LIST*****
    demo PID:13539
    
    *****STATUS FAIL SERVICE LIST*****
    
    $ cmonitor -restart demo
    SUCCESS
    

    clog (集中式日志收集服务)

    实现初衷

    • 实际项目中,服务会部署到多台服务器上去,机器本地日志不方便查看,通过集中收集日志到一台或两台机器上,日志以文件形式存在,按服务名,ip,日期,日志类型分别存储,这样查看日志时就方便多了
    • 我们做服务时,经常需要添加一些跟业务逻辑无关的功能,比如按错误日志报警,上报数据用于统计等等,这些功能和业务逻辑混在一起,实在没有必要,有了clog,我们只需要发送有效的数据,然后就可把数据处理的工作留给clog去做

      功能

    • 通过发送日志至本机agent,然后agent转发至远程master主机,api目前提供golang,c支持
    • 根据配置(master/conf/conf.json)运行相关日志分析程序,目前已实现:日志输出,报警
    • 输出日志文件按master/logs/{模块名}/log{dbg|err|info|war}/{day}/log{ip}{+}{sub}规则命名,最多保存30天日志

    使用方法

    • agent机器

      布署本机agent服务:agent/agent,配置文件:agent/conf/conf.json

    • master机器

      布署master服务:master/master,配置文件:master/conf/conf.json

    • agent和master服务建议用cmonitor启动管理

    注意

    • api.go文件里定义了agent服务端口(agent启动后会监听127.0.0.1:xxx),见clog.Port变量
    • master/conf/conf.json文件里,tpl定义模板,然后通过$xxx方式引用,目前支持的handler有:filehandler和alarmhandler,filehandler用来记录本地日志,alarmhandler用来发报警
    • 对于alarmhandler,相关参数配置见params,目前的报警只是打印日志,实际实用,应替换成自己的报警处理逻辑,重新赋值procs.AlarmFunc就可以了,可以在master/procs目录下新建一个go文件,如下示例:
    package procs
    
    import (
        "encoding/json"
        "os"
    )
    
    func init() {
        // 请替换成你自己的报警处理函数
        AlarmFunc = func(sender string, receivers []string, text string) {
            params := map[string]interface{}{
                "Sender":    sender,
                "Receivers": receivers,
                "Text":      text,
            }
            json.NewEncoder(os.Stdout).Encode(params)
        }
    }
    
    • alarmhandler有防骚扰控制逻辑,相同内容,一分钟内不再报,两次报警不少于30秒,以上限制和日志文件一一对应
    • 如果想添加新的handler,只需在master/procs目录下新建一个go文件,如下示例:
    package procs
    
    func XxxHandler(cate, subcate string, content []byte, params map[string]interface{}) {
    }
    
    func init() {
        RegisterHandler("xxxhandler", XxxHandler)
    }
    

    demo

    api_test.go
    demo (demo项目里有clog的使用例子)

    simplesvr (simple udp server)

    功能:

    • 超简单c/c++服务,多进程,udp通信,没有高深复杂的事件驱动,没有多线程带来的数据共享问题(加锁对性能的影响),代码结构简单,直达业务
    • 适用场景:业务逻辑重,追求高吞吐量,容忍udp带来的不可靠。(已有c lib库,不方便采用golang包装时)
    • c开发新手也可以快速上手

    特性

    • 代码结构简单,仅有一个.cpp文件:main/main.cpp,其它均是.h文件。
    • 调用协议简单,'x00'分隔字段
    • 多进程,同时启动多个业务子进程,任何一个进程(包括父进程)退出,所有其它进程均退出。
    • 支持json格式配置文件
    • 可选通过clog方式记录日志并报警
    • 提供很多有用的小组件,包括: > 简单高效的http get及post操作组件 > 类似go lc的本地缓存组件(支持lru, 支持过期后还能返回旧数据,这个在获取新数据失败时尤其有用)
    • 提供些小的库函数,如:定时器,获取本机内网ip等

      注意

    • 加入新依赖库时,只需要在main/main.cpp里加入库头文件,修改Makefile文件
    • api目录提供api.go示例代码用于和simplesvr服务通信

    gop (go REPL)

    实现初衷

    有时想快速验证go某个函数的使用,临时写个程序太低效,有了gop,立马开一个shell环境,边写边运行,自动为你保存上下文,还可随时导入导出snippet,另外还有代码自动补全等等特性

    特性

    • history record(gop启动后会在home目录下生成.gop文件夹, 输入历史会记录在此)
    • tab complete,可以补全package,补全库函数,需要系统安装有gocode
    • r|w两种模式切换,r是默认模式,对用户输入实时解析运行,执行w命令切换到w模式,w模式下,只有当执行run命令时,代码才会真正执行
    • 代码实时查看和编辑功能[!命令功能]
    • snippet,可以导入和导出模板[<,>命令功能]

      注意:

    • 输入代码时,支持续行
    • 对于如下代码,只会在执行结束后一并输出 > print(1);time.Sleep(time.Second);print(2)
    • 可以通过echo 123这种方式输出, echo是println的简写,你甚至可以重新定义println变量来使用自己的打印方法,比如像我这样定义(utils.IprintD的特点是可以打印出指针指向的实际内容,就算是嵌套的指针也可以,fmt.Printf做不到):
      import "github.com/simplejia/utils"
      var println = utils.IprintD
      
    • 导入项目package时,最好提前通过go install方式安装包文件到pkg目录,这样可以加快执行速度
    • 可以提前import包,后续使用时再自动引入
    • gop启动后会自动导入$PWD/gop.tmpl或者$HOME/.gop/gop.tmpl模板代码,可以把常用的代码保存到gop.tmpl里

    demo

    $ gop
    Welcome to the Go Partner! [[version: 1.7, created by simplejia]
    Enter '?' for a list of commands.
    [r]$ ?
    Commands:
            ?|help  help menu
            -[dpc][#],[#]-[#],...   pop last/specific (declaration|package|code)
            ![!]    inspect source [with linenum]
            <tmpl   source tmpl
            >tmpl   write tmpl
            [#](...)        add def or code
            run     run source
            compile compile source
            w       write source mode on
            r       write source mode off
            reset   reset
            list    tmpl list
    [r]$ for i:=1; i<3; i++ {
    .....    print(i)
    .....    time.Sleep(time.Millisecond)
    .....}
    1
    2
    [r]$ import _ "github.com/simplejia/wsp/demo/mysql"
    [r]$ import _ "github.com/simplejia/wsp/demo/redis"
    [r]$ import _ "github.com/simplejia/wsp/demo/conf"
    [r]$ import "github.com/simplejia/lc"
    [r]$ import "github.com/simplejia/wsp/demo/service"
    [r]$ lc.Init(1024)
    [r]$ demoService := service.NewDemo()
    [r]$ demoService.Set("123", "456")
    [r]$ time.Sleep(time.Millisecond)
    [r]$ echo demoService.Get("123")
    456
    [r]$ >gop
    [r]$ <gop
    [r]$ !
            package main
    
    p0:     import _ "github.com/simplejia/wsp/demo/mysql"
    p1:     import _ "github.com/simplejia/wsp/demo/redis"
    p2:     import _ "github.com/simplejia/wsp/demo/conf"
    p3:     import "github.com/simplejia/lc"
    p4:     import "github.com/simplejia/wsp/demo/service"
    p5:     import "fmt" // imported and not used
    p6:     import "strconv" // imported and not used
    p7:     import "strings" // imported and not used
    p8:     import "time" // imported and not used
    p9:     import "encoding/json" // imported and not used
    p10:    import "bytes" // imported and not used
    
            func main() {
    c0:             lc.Init(1024)
    c1:             demoService := service.NewDemo()
    c2:             _ = demoService
    c3:             demoService.Set("123", "456")
    c4:             time.Sleep(time.Millisecond)
            }
    
    [r]$
  • 相关阅读:
    pt-tcp-model
    (转)从史上八大MySQL宕机事故中学到的经验
    pt-query-digest
    DNS生效时间
    Python之uuid模块
    一个IO的传奇一生
    Python之Queue模块
    利用freemarker 静态化网页
    FreeMarker教程
    模板引擎freemarker的简单使用教程
  • 原文地址:https://www.cnblogs.com/purpleraintear/p/6046454.html
Copyright © 2011-2022 走看看