go-zero单体服务使用泛型简化注册Handler路由

一、Golang环境安装及配置Go Module

https://go-zero.dev/cn/docs/prepare/golang-install

mac OS安装Go

  • 下载并安装Go for Mac
  • 验证安装结果
$ go version
go version go1.15.1 darwin/amd64

linux 安装Go

  • 下载Go for Linux
  • 解压压缩包至/usr/local
$ tar -C /usr/local -xzf go1.15.8.linux-amd64.tar.gz
  • 添加/usr/local/go/bin到环境变量
$ $HOME/.profile
$ export PATH=$PATH:/usr/local/go/bin
$ source $HOME/.profile
  • 验证安装结果
$ go version
go version go1.15.1 linux/amd64

Windows安装Go

  • 下载并安装Go for Windows
  • 验证安装结果
$ go version
go version go1.15.1 windows/amd64

MODULE配置

Go Module是Golang管理依赖性的方式,像Java中的Maven,Android中的Gradle类似。

  • 查看GO111MODULE开启情况
$ go env GO111MODULE
on
  • 开启GO111MODULE,如果已开启(即执行go env GO111MODULE结果为on)请跳过。
$ go env -w GO111MODULE="on"
  • 设置GOPROXY
$ go env -w GOPROXY=https://goproxy.cn
  • 设置GOMODCACHE
查看GOMODCACHE

$ go env GOMODCACHE

  • 如果目录不为空或者/dev/null,请跳过。
go env -w GOMODCACHE=$GOPATH/pkg/mod

二、Goctl 安装

Goctl在go-zero项目开发着有着很大的作用,其可以有效的帮助开发者大大提高开发效率,减少代码的出错率,缩短业务开发的工作量,更多的Goctl的介绍请阅读Goctl介绍

  • 安装(mac&linux)
### Go 1.15 及之前版本
GO111MODULE=on GOPROXY=https://goproxy.cn/,direct go get -u github.com/zeromicro/go-zero/tools/goctl@latest

### Go 1.16 及以后版本
GOPROXY=https://goproxy.cn/,direct go install github.com/zeromicro/go-zero/tools/goctl@latest
  • 安装(windows)
go install github.com/zeromicro/go-zero/tools/goctl@latest
  • 环境变量检测(mac&linux)
    go get 下载编译后的二进制文件位于 \$GOPATH/bin 目录下,要确保 $GOPATH/bin已经添加到环境变量。
sudo vim /etc/paths //添加环境变量

在最后一行添加如下内容 //$GOPATH 为你本机上的文件地址

$GOPATH/bin
  • 安装结果验证
$ goctl -v
goctl version 1.1.4 darwin/amd64

二、初始化go-zero

goctl api new greet
cd greet
go mod init
go mod tidy
go run greet.go -f etc/greet-api.yaml
  • 默认侦听在 8888 端口
    侦听端口可以在 greet-api.yaml配置文件里修改,此时,可以通过 curl 请求,或者直接在浏览器中打开 http://localhost:8888/from/you
$ curl -i http://localhost:8888/from/you

HTTP/1.1 200 OK
Content-Type: application/json; charset=utf-8
Traceparent: 00-45fa9e7a7c505bad3a53a024e425ace9-eb5787234cf3e308-00
Date: Thu, 22 Oct 2020 14:03:18 GMT
Content-Length: 14

null
  • greet服务的目录结构
$ tree greet
greet
├── etc
│   └── greet-api.yaml
├── greet.api
├── greet.go
└── internal
    ├── config
    │   └── config.go
    ├── handler
    │   ├── greethandler.go
    │   └── routes.go
    ├── logic
    │   └── greetlogic.go
    ├── svc
    │   └── servicecontext.go
    └── types
        └── types.go

三、查看注册Handler路由流程

  • greet.go
var configFile = flag.String("f", "etc/greet-api.yaml", "the config file")

func main() {
    flag.Parse()

    var c config.Config
    conf.MustLoad(*configFile, &c)

    server := rest.MustNewServer(c.RestConf)
    defer server.Stop()
        //上面的都是加载配置什么的
    ctx := svc.NewServiceContext(c)
    handler.RegisterHandlers(server, ctx) //此方法是注册路由和路由映射Handler,重点在这里

    fmt.Printf("Starting server at %s:%d...\n", c.Host, c.Port)
    server.Start()
}
  • RegisterHandlers在 internal\handler\routes.go
func RegisterHandlers(server *rest.Server, serverCtx *svc.ServiceContext) {
   server.AddRoutes( //往rest.Server中添加路由
    []rest.Route{ //路由数组
       {
          Method:  http.MethodGet,
          Path:    "/from/:name", //路由
          Handler: GreetHandler(serverCtx),//当前路由的处理Handler
       },
    },
   )
}
  • GreetHandler在 internal\handler\greethandler.go
func GreetHandler(ctx *svc.ServiceContext) http.HandlerFunc {
    return func(w http.ResponseWriter, r *http.Request) {
1.      var req types.Request
2.      if err := httpx.Parse(r, &req); err != nil { //请求的错误判断,这个可以不用管
3.          httpx.Error(w, err)
4.          return
5.      }

        l := logic.NewGreetLogic(r.Context(), ctx) //GreetHandler处理函数将请求转发到了GreetLogic中,调用NewGreetLogic进行结构体的初始化
        resp, err := l.Greet(req) //然后调用Greet来进行处理请求,所以我们在GreetLogic.Greet方法中可以看到一句话// todo: add your logic here and delete this line
        if err != nil {
            httpx.Error(w, err)
        } else {
            httpx.OkJson(w, resp)
        }
    }
}

四、对注册Handler路由进行简化

项目文件的增加

在路由注册时,我们如果服务越加越多,那么相对应的 func xxxxHandler(ctx *svc.ServiceContext) http.HandlerFunc就要进行多次的添加,并且这个方法体内部1到5行是属于额外的重复添加
例如:我们添加一个 customlogic.go
按照命名的正确和规范性,需要在 internal\logic目录下添加customlogic.go文件,然后在 internal\handler目录下添加customhandler.go文件,并且两个文件都添加相对应的结构体和函数等,最后在 routes.go中再添加一次

{
    Method:  http.MethodGet,
    Path:    "/custom/:name",
    Handler: CustomHandler(serverCtx),
},

此时,我们的文件结构应该是这样

greet
├── etc
│   └── greet-api.yaml
├── greet.api
├── greet.go
└── internal
    ├── config
    │   └── config.go
    ├── handler
    │   ├── greethandler.go
    │   ├── customhandler.go
    │   ├── ...

    │   └── routes.go
    ├── logic
    │   ├── greetlogic.go
    │   ├── ...

    │   └── customlogic.go
    ├── svc
    │   └── servicecontext.go
    └── types
        └── types.go

当单体应用达到一定的数量级,handler和logic文件夹下将会同步增加很多的文件

引入泛型概念

自Go1.18开始,go开始使用泛型,泛型的广泛定义 :是一种把明确类型的工作推迟到创建对象或者调用方法的时候才去明确的特殊的类型。 也就是说在泛型使用过程中,操作的数据类型被指定为一个参数,而这种参数类型可以用在 类、方法和接口 中,分别被称为 泛型类 、 泛型方法 、 泛型接口 。
我们可以利用泛型,让在添加路由时就要固定死的 Handler: GreetHandler(serverCtx)推迟到后面,去根据实际的Logic结构体去判断需要真正执行的 logic.NewGreetLogic(r.Context(), ctx)初始化结构体和 l.Greet(req)逻辑处理方法

如何去做

  1. internal\logic下添加一个 baselogic.go文件,参考Go泛型实战 | 如何在结构体中使用泛型
package logic

import (
    "greet/internal/svc"
    "greet/internal/types"
    "net/http"
)

type BaseLogic interface {
    any
    Handler(req types.Request, w http.ResponseWriter, r *http.Request, svcCtx *svc.ServiceContext) //每一个结构体中必须要继承一下Handler方法,例如customlogic.go和greetlogic.go中的Handler方法
}

type logic[T BaseLogic] struct {
    data T
}

func New[T BaseLogic]() logic[T] {
    c := logic[T]{}
    var ins T
    c.data = ins
    return c
}
func (a *logic[T]) LogicHandler(req types.Request, w http.ResponseWriter, r *http.Request, svcCtx *svc.ServiceContext) { //作为一个中转处理方法,最终执行结构体的Handler
    a.data.Handler(req, w, r, svcCtx)
}
  1. greethandler.go文件修改成 basehandler.go,注释掉之前的 GreetHandler方法
package handler

import (
    "net/http"

    "greet/internal/logic"
    "greet/internal/svc"
    "greet/internal/types"

    "github.com/zeromicro/go-zero/rest/httpx"
)

// func GreetHandler(svcCtx *svc.ServiceContext) http.HandlerFunc {
//  return BaseHandlerFunc(svcCtx)
//  // return func(w http.ResponseWriter, r *http.Request) {
//  //  var req types.Request
//  //  if err := httpx.Parse(r, &req); err != nil {
//  //      httpx.Error(w, err)
//  //      return
//  //  }
//  //  l := logic.NewGreetLogic(r.Context(), svcCtx)
//  //  resp, err := l.Greet(&req)
//  //  if err != nil {
//  //      httpx.Error(w, err)
//  //  } else {
//  //      httpx.OkJson(w, resp)
//  //  }
//  // }
// }

func BaseHandlerFunc[T logic.BaseLogic](svcCtx *svc.ServiceContext, t T) http.HandlerFunc {
    return func(w http.ResponseWriter, r *http.Request) {
        var req types.Request
        if err := httpx.Parse(r, &req); err != nil {
            httpx.Error(w, err)
            return
        }
        //通过泛型动态调用不同结构体的Handler方法
        cc := logic.New[T]()
        cc.LogicHandler(req, w, r, svcCtx)
    }
}
  1. internal\logic\greetlogic.go中增加一个 Handler方法
package logic

import (
    "context"
    "net/http"

    "greet/internal/svc"
    "greet/internal/types"

    "github.com/zeromicro/go-zero/core/logx"
    "github.com/zeromicro/go-zero/rest/httpx"
)

type GreetLogic struct {
    logx.Logger
    ctx    context.Context
    svcCtx *svc.ServiceContext
}

func NewGreetLogic(ctx context.Context, svcCtx *svc.ServiceContext) *GreetLogic {
    return &GreetLogic{
        Logger: logx.WithContext(ctx),
        ctx:    ctx,
        svcCtx: svcCtx,
    }
}
func (a GreetLogic) Handler(req types.Request, w http.ResponseWriter, r *http.Request, svcCtx *svc.ServiceContext) { //新增方法
    l := NewGreetLogic(r.Context(), svcCtx)
    resp, err := l.Greet(&req)
    if err != nil {
        httpx.Error(w, err)
    } else {
        httpx.OkJson(w, resp)
    }
}

func (l *GreetLogic) Greet(req *types.Request) (resp *types.Response, err error) {
    // todo: add your logic here and delete this line
    response := new(types.Response)
    if (*req).Name == "me" {
        response.Message = "greetLogic: listen to me, thank you."
    } else {
        response.Message = "greetLogic: listen to you, thank me."
    }

    return response, nil
}
  1. 然后修改 internal\handler\routes.go下面的 server.AddRoutes部分
func RegisterHandlers(server *rest.Server, serverCtx *svc.ServiceContext) {
   server.AddRoutes( //往rest.Server中添加路由
    []rest.Route{ //路由数组
       {
          Method:  http.MethodGet,
          Path:    "/from/:name", //路由
          Handler: BaseHandlerFunc(serverCtx,logic.GreetLogic{}),
       },
    },
   )
}

现在就大功告成了,我们启动一下

go run greet.go -f etc/greet-api.yaml

然后在浏览器中请求一下 http://localhost:8888/from/you

go-zero单体服务使用泛型简化注册Handler路由

验证一下新增api路由

  1. internal\logic下新增一个 customlogic.go文件
package logic

import (
    "context"
    "net/http"

    "greet/internal/svc"
    "greet/internal/types"

    "github.com/zeromicro/go-zero/core/logx"
    "github.com/zeromicro/go-zero/rest/httpx"
)

type CustomLogic struct {
    logx.Logger
    ctx    context.Context
    svcCtx *svc.ServiceContext
}

func NewCustomLogic(ctx context.Context, svcCtx *svc.ServiceContext) *CustomLogic {
    return &CustomLogic{
        Logger: logx.WithContext(ctx),
        ctx:    ctx,
        svcCtx: svcCtx,
    }
}

func (a CustomLogic) Handler(req types.Request, w http.ResponseWriter, r *http.Request, svcCtx *svc.ServiceContext) {
    l := NewCustomLogic(r.Context(), svcCtx)
    resp, err := l.Custom(&req)
    if err != nil {
        httpx.Error(w, err)
    } else {
        httpx.OkJson(w, resp)
    }
}

func (l *CustomLogic) Custom(req *types.Request) (resp *types.Response, err error) { //response.Message稍微修改了一下,便于区分
    // todo: add your logic here and delete this line
    response := new(types.Response)
    if (*req).Name == "me" {
        response.Message = "customLogic: listen to me, thank you."
    } else {
        response.Message = "customLogic: listen to you, thank me."
    }

    return response, nil
}
  1. 然后修改 internal\handler\routes.go
func RegisterHandlers(server *rest.Server, serverCtx *svc.ServiceContext) {
   server.AddRoutes( //往rest.Server中添加路由
    []rest.Route{ //路由数组
       {
          Method:  http.MethodGet,
          Path:    "/from/:name", //路由
          Handler: BaseHandlerFunc(serverCtx,logic.GreetLogic{}),
       },
           {
          Method:  http.MethodGet,
          Path:    "/to/:name", //路由
          Handler: BaseHandlerFunc(serverCtx,logic.CustomLogic{}),
       },
    },
   )
}

其他地方不需要修改
我们启动一下

go run greet.go -f etc/greet-api.yaml

然后在浏览器中请求一下 http://localhost:8888/from/youhttp://localhost:8888/to/youhttp://localhost:8888/too/you

go-zero单体服务使用泛型简化注册Handler路由

现在,在添加新的logic做路由映射时,就可以直接简化掉添加 xxxxhandler.go文件了,实际上是将这个Handler移动到了xxxxlogic.go中。

新手,不喜轻喷

五、后面对于之前繁琐的泛型使用进行了简化

BaseHandlerFunc中改为

func BaseHandlerFunc[T logic.BaseLogic](svcCtx *svc.ServiceContext, t T) http.HandlerFunc {
    return func(w http.ResponseWriter, r *http.Request) {
        var req types.Request
        if err := httpx.Parse(r, &req); err != nil {
            httpx.Error(w, err)
            return
        }
        var ins T
        ins.Handler(req, w, r, svcCtx)
        //通过泛型动态调用不同结构体的Handler方法
        // cc := logic.New[T]()
        // cc.LogicHandler(req, w, r, svcCtx)
    }
}

baselogic.go改为

package logic
 import (
 "greet/internal/svc"
 "greet/internal/types"
 "net/http"
 )
 type BaseLogic interface {
   any
   Handler(req types.Request, w http.ResponseWriter, r *http.Request, svcCtx *svc.ServiceContext) //每一个结构体中必须要继承一下Handler方法,例如customlogic.go和greetlogic.go中的Handler方法
 }
   // type logic[T BaseLogic] struct {
 // data T
 // }
   // func New[T BaseLogic]() logic[T] {
 // c := logic[T]{}
 // var ins T
 // c.data = ins
 // return c
 // }
 // func (a *logic[T]) LogicHandler(req types.Request, w http.ResponseWriter, r *http.Request, svcCtx *svc.ServiceContext) {
 // a.data.Handler(req, w, r, svcCtx)
 // }

这样就可以了

Original: https://www.cnblogs.com/spatxos/p/16524742.html
Author: spatxos
Title: go-zero单体服务使用泛型简化注册Handler路由

原创文章受到原创版权保护。转载请注明出处:https://www.johngo689.com/516241/

转载文章受原作者版权保护。转载请注明原作者出处!

(0)

大家都在看

  • Go语言 异常panic和恢复recover用法

    背景:Go语言追求简洁优雅,所以,Go语言不支持传统的 try…catch…finally 这种异常,因为Go语言的设计者们认为,将异常与控制结构混在一起会…

    Go语言 2023年5月29日
    063
  • 紫色飞猪的研发之旅–06go自定义状态码

    在实际开发中,需要前后端需要协商状态码,状态码用于后端返前端时使用。在一个团队中,定义的状态码讲道理应该是一致的,项目开始的起始阶段状态码应该是定义了个七七八八的,随着功能的叠加而…

    Go语言 2023年5月25日
    079
  • Go切片全解析

    目录结构:数组切片 底层结构 创建 普通声明 make方式 截取 边界问题 追加 拓展表达式 扩容机制 切片传递的坑 切片的拷贝 浅拷贝 深拷贝 var n [4]int fmt….

    Go语言 2023年5月25日
    091
  • golang实现一个简单的websocket聊天室

    基本原理:1.引入了 golang.org/x/net/websocket 包。2.监听端口。3.客户端连接时,发送结构体: {“type”:”…

    Go语言 2023年5月25日
    083
  • Go 语言快速开发入门

    需求 开发的步骤 linux下如何开发Go程序 MAC下如何开发Go程序 Golang执行流程分析 编译和运行说明 Go程序开发的注意事项 Go语言的转义字符(escapechar…

    Go语言 2023年5月25日
    084
  • go微服务框架Kratos笔记(六)链路追踪实战

    什么是链路追踪 借用阿里云链路追踪文档来解释分布式链路追踪(Distributed Tracing),也叫 分布式链路跟踪,分布式跟踪,分布式追踪 等等,它为分布式应用的开发者提供…

    Go语言 2023年5月25日
    073
  • Go 的 golang.org/x/ 系列包和标准库包有什么区别?

    在开发过程中可能会遇到这样的情况,有一些包是引入自不同地方的,比如: golang.org/x/net/html 和 net/html, golang.org/x/crypto 和…

    Go语言 2023年5月25日
    074
  • Go语言实践模式 – 函数选项模式(Functional Options Pattern)

    什么是函数选项模式 大家好,我是小白,有点黑的那个白。 最近遇到一个问题,因为业务需求,需要对接三方平台. 而三方平台提供的一些HTTP(S)接口都有统一的密钥生成规则要求. 为此…

    Go语言 2023年5月25日
    057
  • go语言 函数return值的几种情况

    分三种情况 (以下 “指定返回值”这句话, 仅指return后面直接跟着的返回值) 退出执行,不指定返回值 *(1) 函数没有返回值 package mai…

    Go语言 2023年5月29日
    096
  • Go语言之变量与基础数据类型

    Go 是静态(编译型)语言,是区别于解释型语言的弱类型语言(静态:类型固定,强类型:不同类型不允许直接运算) 例如 python 就是动态强类型语言 1、Go 的特性: 跨平台的编…

    Go语言 2023年5月25日
    069
  • sqlx操作MySQL实战及其ORM原理

    sqlx是Golang中的一个知名三方库,其为Go标准库database/sql提供了一组扩展支持。使用它可以方便的在数据行与Golang的结构体、映射和切片之间进行转换,从这个角…

    Go语言 2023年5月25日
    079
  • 基于LSM的Key-Value数据库实现稀疏索引篇

    上篇文章简单的填了一个坑基于LSM数据库的实现了WAL,在该版本中如数据写入到内存表的同时将未持久化的数据写入到WAL文件,在未将数据持久化时程序崩溃,可通过WAL文件将数据还原恢…

    Go语言 2023年5月25日
    089
  • go语言学习笔记-初识Go语言

    Go语言是怎样诞生的? Go语言的创始人有三位,分别是图灵奖获得者、C语法联合发明人、Unix之父肯·汤普森(Ken Thompson)、Plan 9操作系统领导者、UTF-8编码…

    Go语言 2023年5月25日
    089
  • 微服务追踪SQL(支持Isto管控下的gorm查询追踪)

    效果图 SQL的追踪正确插入到微服务的调用链之间 详细记录了SQL的执行内容和消耗时间 搜索SQL的类型 多线程(goroutine)下的追踪效果 在 Kubernetes 中部署…

    Go语言 2023年5月25日
    084
  • Go之Logrus用法入门

    Logrus是Go (golang)的结构化日志程序,完全兼容标准库的API日志程序。Logrus is a structured logger for Go (golang), …

    Go语言 2023年5月25日
    080
  • AES加解密(golang <--> crypto-js)

    AES(Advanced Encryption Standard) 是一种对称加密算法,是比 DES 更好的对称加密算法类。 使用AES,在前后端之间传送密码等相关数据时,能简单高…

    Go语言 2023年5月25日
    056
亲爱的 Coder【最近整理,可免费获取】👉 最新必读书单  | 👏 面试题下载  | 🌎 免费的AI知识星球