关于go-zero单体服务使用泛型简化注册Handler路由的问题
知识点掌握了,还需要不断练习才能熟练运用。下面golang学习网给大家带来一个Golang开发实战,手把手教大家学习《关于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) { var req types.Request if err := httpx.Parse(r, &req); err != nil { //请求的错误判断,这个可以不用管 httpx.Error(w, err) return } 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)
逻辑处理方法
如何去做在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) }
将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) } }
在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 }
然后修改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
验证一下新增api路由在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 }
然后修改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/you
、http://localhost:8888/to/you
、http://localhost:8888/too/you
现在,在添加新的logic做路由映射时,就可以直接简化掉添加xxxxhandler.go
文件了,实际上是将这个Handler移动到了xxxxlogic.go中。
本文代码放在go-zero-monolithic-service-generics
终于介绍完啦!小伙伴们,这篇关于《关于go-zero单体服务使用泛型简化注册Handler路由的问题》的介绍应该让你收获多多了吧!欢迎大家收藏或分享给更多需要学习的朋友吧~golang学习网公众号也会发布Golang相关知识,快来关注吧!

- 上一篇
- Go泛型实战教程之如何在结构体中使用泛型

- 下一篇
- go语言VScode see 'go help modules' (exit status 1)问题的解决过程
-
- 独特的胡萝卜
- 这篇技术贴真是及时雨啊,好细啊,真优秀,mark,关注师傅了!希望师傅能多写Golang相关的文章。
- 2023-01-27 14:46:06
-
- 听话的鞋垫
- 写的不错,一直没懂这个问题,但其实工作中常常有遇到...不过今天到这,看完之后很有帮助,总算是懂了,感谢楼主分享文章!
- 2023-01-03 07:36:31
-
- Golang · Go教程 | 28分钟前 |
- DebianSyslog在虚拟机中的实用攻略
- 467浏览 收藏
-
- Golang · Go教程 | 8小时前 |
- DebianOpenSSL安装失败的终极解决方案
- 501浏览 收藏
-
- Golang · Go教程 | 9小时前 |
- Debian数据快速提取技巧
- 216浏览 收藏
-
- Golang · Go教程 | 12小时前 |
- Debian系统JS依赖管理终极攻略
- 218浏览 收藏
-
- Golang · Go教程 | 14小时前 |
- Debian上Hadoop作业调度实用技巧
- 100浏览 收藏
-
- Golang · Go教程 | 14小时前 |
- Go语言闭包误区与匿名函数深度解析
- 222浏览 收藏
-
- Golang · Go教程 | 14小时前 |
- Debian系统安全回收数据的正确攻略
- 111浏览 收藏
-
- Golang · Go教程 | 16小时前 |
- Debian高效fetch技巧与使用攻略
- 125浏览 收藏
-
- 前端进阶之JavaScript设计模式
- 设计模式是开发人员在软件开发过程中面临一般问题时的解决方案,代表了最佳的实践。本课程的主打内容包括JS常见设计模式以及具体应用场景,打造一站式知识长龙服务,适合有JS基础的同学学习。
- 542次学习
-
- GO语言核心编程课程
- 本课程采用真实案例,全面具体可落地,从理论到实践,一步一步将GO核心编程技术、编程思想、底层实现融会贯通,使学习者贴近时代脉搏,做IT互联网时代的弄潮儿。
- 508次学习
-
- 简单聊聊mysql8与网络通信
- 如有问题加微信:Le-studyg;在课程中,我们将首先介绍MySQL8的新特性,包括性能优化、安全增强、新数据类型等,帮助学生快速熟悉MySQL8的最新功能。接着,我们将深入解析MySQL的网络通信机制,包括协议、连接管理、数据传输等,让
- 497次学习
-
- JavaScript正则表达式基础与实战
- 在任何一门编程语言中,正则表达式,都是一项重要的知识,它提供了高效的字符串匹配与捕获机制,可以极大的简化程序设计。
- 487次学习
-
- 从零制作响应式网站—Grid布局
- 本系列教程将展示从零制作一个假想的网络科技公司官网,分为导航,轮播,关于我们,成功案例,服务流程,团队介绍,数据部分,公司动态,底部信息等内容区块。网站整体采用CSSGrid布局,支持响应式,有流畅过渡和展现动画。
- 484次学习
-
- 笔灵AI生成答辩PPT
- 探索笔灵AI生成答辩PPT的强大功能,快速制作高质量答辩PPT。精准内容提取、多样模板匹配、数据可视化、配套自述稿生成,让您的学术和职场展示更加专业与高效。
- 15次使用
-
- 知网AIGC检测服务系统
- 知网AIGC检测服务系统,专注于检测学术文本中的疑似AI生成内容。依托知网海量高质量文献资源,结合先进的“知识增强AIGC检测技术”,系统能够从语言模式和语义逻辑两方面精准识别AI生成内容,适用于学术研究、教育和企业领域,确保文本的真实性和原创性。
- 24次使用
-
- AIGC检测-Aibiye
- AIbiye官网推出的AIGC检测服务,专注于检测ChatGPT、Gemini、Claude等AIGC工具生成的文本,帮助用户确保论文的原创性和学术规范。支持txt和doc(x)格式,检测范围为论文正文,提供高准确性和便捷的用户体验。
- 30次使用
-
- 易笔AI论文
- 易笔AI论文平台提供自动写作、格式校对、查重检测等功能,支持多种学术领域的论文生成。价格优惠,界面友好,操作简便,适用于学术研究者、学生及论文辅导机构。
- 42次使用
-
- 笔启AI论文写作平台
- 笔启AI论文写作平台提供多类型论文生成服务,支持多语言写作,满足学术研究者、学生和职场人士的需求。平台采用AI 4.0版本,确保论文质量和原创性,并提供查重保障和隐私保护。
- 35次使用
-
- gozero微服务框架logx日志组件剖析
- 2022-12-24 314浏览
-
- Go泛型实战教程之如何在结构体中使用泛型
- 2023-02-24 234浏览
-
- Go1.18新特性工作区模糊测试及泛型的使用详解
- 2022-12-30 131浏览
-
- Go泛型应用工厂方法及泛型使用
- 2022-12-27 185浏览
-
- go zero微服务实战性能优化极致秒杀
- 2022-12-27 207浏览