快速学习go-zero

go的web框架有很多,目前go的社区大家对于框架的态度也不尽相同,有些轻量级的框架,但是也就代表整合第三方中间件就需要自己根据客户端进行封装,比如gin+gorm,也有些功能完全但是被认为丢失了go本身轻量设计的初衷,
比如goframe,而同样的微服务有很多框架,国内比较出门的就是go-zero ,有专门的开发工具goctl,让开发者只需要关注业务代码即可完成微服务的上线。

介绍

go-zero 是一个集成了各种工程实践的 web 和 rpc 框架。通过弹性设计保障了大并发服务端的稳定性,经受了充分的实战检验。

go-zero 包含极简的 API 定义和生成工具 goctl,可以根据定义的 api 文件一键生成 Go, iOS, Android, Kotlin, Dart, TypeScript, JavaScript 代码,并可直接运行。

开发者只需要编写业务代码 就可以完成微服务的构建
红色代表需要开发者手写的部分!
在这里插入图片描述

所以重点是通过goctl 来根据编写的api文件和proto文件快速生成代码,让大部分时间只关心业务,缩短时间成本,这一点使用起来比goframe成熟一些(个人感觉),写下该笔记也是因为现在看文档的时候觉得go-zero 没有之前的文档那样对新手友好了

安装教程

安装官方脚手架 go 从ctroller

#官方脚手架
go install github.com/zeromicro/go-zero/tools/goctl@latest
#protobuf 工具
goctl env check --install --verbose --force
#框架
go get -u github.com/zeromicro/go-zero@latest

验证版本

goctl -version

快速入门

web模块

//生成 api web模块 目录greet
goctl api new greet
cd greet
go mod init
go mod tidy
go run greet.go -f etc/greet-api.yaml

此时访问localhost:8888就会得到一个null 但是控制器有对应的输出日志

根据api文件 生成web服务 官方的快速入门案列 zero-doc/doc/shorturl.md at main · zeromicro/zero-doc (github.com)

#生成一个基本的api文件 api是go-zero的一个文件 用于goctl来快速的生成web代码
goctl api -o shorturl.api

替换内容为

type (
  expandReq {
    shorten string `form:"shorten"`
  }

  expandResp {
    url string `json:"url"`
  }
)

type (
  shortenReq {
    url string `form:"url"`
  }

  shortenResp {
    shorten string `json:"shorten"`
  }
)

service shorturl-api {
  @handler ShortenHandler
  get /shorten(shortenReq) returns(shortenResp)

  @handler ExpandHandler
  get /expand(expandReq) returns(expandResp)
}
  • service shorturl-api { 这一行定义了 service 名字

  • @server 部分用来定义 server 端用到的属性

  • handler 定义了服务端 handler 名字

  • get /shorten(shortenReq) returns(shortenResp) 定义了 get 方法的路由、请求参数、返回参数等

  • type生成的交互结构体也会在对应文件夹

感觉和proto文件差不多 也是修改该文件快速生成代码

进行该目录 根据api文件 生成代码

goctl api go -api shorturl.api -dir .

后续如果数据结构以及更新接口也是编辑api文件,编辑生成的go文件会报错不应该生成源文件

goctl api go -api order.api -dir .

并且不会覆盖已经写好的逻辑

etc/web-api.yaml exists, ignored generation
internal/config/config.go exists, ignored generation
web.go exists, ignored generation
internal/svc/servicecontext.go exists, ignored generation
internal/handler/webhandler.go exists, ignored generation
internal/handler/orderhandler.go exists, ignored generation
internal/logic/weblogic.go exists, ignored generation
internal/logic/orderlogic.go exists, ignored generation
Done.

生成的web目录结构

.
├── api
│ ├── etc
│ │ └── shorturl-api.yaml // 配置文件
│ ├── internal
│ │ ├── config
│ │ │ └── config.go // 定义配置
│ │ ├── handler
│ │ │ ├── expandhandler.go // 实现 expandHandler
│ │ │ ├── routes.go // 定义路由处理
│ │ │ └── shortenhandler.go // 实现 shortenHandler
│ │ ├── logic
│ │ │ ├── expandlogic.go // 实现 ExpandLogic
│ │ │ └── shortenlogic.go // 实现 ShortenLogic
│ │ ├── svc
│ │ │ └── servicecontext.go // 定义 ServiceContext
│ │ └── types
│ │ └── types.go // 定义请求、返回结构体
│ ├── shorturl.api
│ └── shorturl.go // main 入口定义
├── go.mod
└── go.sum

在这里插入图片描述

  • type存放api生成的接口响应值和请求值

  • svc上下文变量 在rpc模块的时候需要经常用到

  • logic 是逻辑实现模块 类似mvc中接口的实现类

  • handler为路由注册,一个路由/XXX/XX 对应一个handler

    • 任意打开生成的一个handler

      在这里插入图片描述

      和gin goframe一样也是对原生http请求进行封装 其中调用logic把开发者的逻辑和返回值写回http响应体

  • config 运行时候的上下文配置

运行

go run shorturl.go -f etc/shorturl.yaml

启动类解析代码

#如果把这里的目录地址换成项目的引用地址
var configFile = flag.String("f", "shorturl/api/etc/shorturl.yaml", "the config file")
#作用是自定义指令 如果没有使用-f指定配置文件 则默认是第二个参数位置
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)
    fmt.Printf("Starting server at %s:%d...\n", c.Host, c.Port)
    server.Start()
}

就可以不用写-f参数 在idea运行

生成web模块的其他语言代码

goctl api java -api greet.api -dir greet
goctl api dart -api greet.api -dir greet

官网框架概述 | go-zero Documentation

rpc协议采用的是grpc 需要根据官网安装工具

rpc模块

新建一个目录

在 shorturl 目录下创建 rpc/transform 目录

rpc/transform

在 rpc/transform 目录下编写 transform.proto 文件

可以通过命令生成 proto 文件模板

goctl rpc -o transform.proto

修改后文件将内容替换如下:

syntax = "proto3";

package transform;

option go_package = "./transform";

message expandReq{
  string shorten = 1;
}

message expandResp{
  string url = 1;
}

message shortenReq{
  string url = 1;
}

message shortenResp{
  string shorten = 1;
}

service  transformer{
  rpc expand(expandReq) returns(expandResp);
  rpc shorten(shortenReq) returns(shortenResp);
}

用 goctl 生成 rpc 代码,在 rpc/transform 目录下执行命令

goctl rpc protoc student.proto --go_out=. --go-grpc_out=. --zrpc_out=.

如果后续更新接口 响应体信息 生成代码单不覆盖已经写好的逻辑结构 支付该rpc 结构部分

 protoc --go_out=. --go-grpc_out=. order.proto

注意:不能在 GOPATH 目录下执行以上命令

文件结构如下:

rpc/transform
├── etc
│ └── transform.yaml // 配置文件
├── internal
│ ├── config
│ │ └── config.go // 配置定义
│ ├── logic
│ │ ├── expandlogic.go // expand 业务逻辑在这里实现
│ │ └── shortenlogic.go // shorten 业务逻辑在这里实现
│ ├── server
│ │ └── transformerserver.go // 调用入口, 不需要修改
│ └── svc
│ └── servicecontext.go // 定义 ServiceContext,传递依赖
├── transform
│ ├── transform.pb.go
│ └── transform_grpc.pb.go
├── transform.go // rpc 服务 main 函数
├── transform.proto
└── transformer
└── transformer.go // 提供了外部调用方法,无需修改
执行 go mod tidy 整理依赖

启动 etcd server (已经安装好)

etcd

启动 rpc 服务直接可以运行,如下:

go run transform.go -f etc/transform.yaml

Starting rpc server at 127.0.0.1:888…
查看服务是否注册,以下值为参考值,主要观察 etcd 有注册到 transform.rpc 的 key 和 8080 端口即可,各自机器的 ip 结果不一样。

etcdctl get transform.rpc --prefix
#PS C:\Users\侯> etcdctl get transform.rpc --prefix
#transform.rpc/7587881007321565706
#169.254.67.206:888
目录文件解析

在这里插入图片描述

大致和api web项目的目录差不多 区别在于没有handler

和pro文件名一样的是goctl生成服务端代码

在这里插入图片描述

logic用来实现

如果后续需要抛出的接口很多 就续写proto文件即可

 protoc --go_out=. --go-grpc_out=. student.proto   

因为rpc注册到etcd的时,服务之间内部采用key通信

在这里插入图片描述

模拟真实场景

现在为了模拟一个真实场景 有以下学生商城的业务需求 用户向网关发送请求(采用普通web服务模拟)传递订单id ,网关转发该请求到order,order拿到该id以后向学生服务发起请求获取学生info的业务场景

用户->网关(web)->orderservice->studentservice 这样的微服务查询

如果是在java的那一套,编写配置类,编写接口,响应体,请求体需要的体量就大一些,go-zero使用api文件和grpc(proto)文件完成微服务的快速编写

目录结构

在这里插入图片描述

一个网关 一个学生服务 一个订单

编写student-service

我个人的习惯对于需求的链式调用是从底层写到高层

student.proto

syntax = "proto3";
//指定生成的proto部分文件输出路径
option go_package="./student";
package student;

service StudentService {
  rpc GetStudentInfo (StudentRequest) returns (StudentResponse);
}

message StudentRequest {
  int32 student_id = 1;
}

message StudentResponse {
  int32 student_id = 1;
  string name = 2;
  int32 age = 3;
}


定义了学生服务 ,该服务只有一个接口获取学生个人信息

goctl rpc protoc student.proto --go_out=. --go-grpc_out=. --zrpc_out=.

生成项目模板后 对grpc 只要熟悉的开发者就可以找到 在指定的输出路径有 生成的grpc服务和客户端

在这里插入图片描述

点击idea的实现提示就会跳转到 internal 目录下的server目录 ,而其中就包含 logic实列化来处理

func (s *StudentServiceServer) GetStudentInfo(ctx context.Context, in *student.StudentRequest) (*student.StudentResponse, error) {
    l := logic.NewGetStudentInfoLogic(ctx, s.svcCtx)
    return l.GetStudentInfo(in)
}

所以我们需要重写的就是生成在logic获取学生信息接口

模拟数据中,只要由id为1就可以成功调用

func (l *GetStudentInfoLogic) GetStudentInfo(req *student.StudentRequest) (*student.StudentResponse, error) {
	fmt.Println("学生服务被调用")
	fmt.Printf("得到id%d\n", req.StudentId)
	if req.StudentId == 1 {
		return &student.StudentResponse{
			StudentId: req.StudentId,
			Name:      "坏学生乔治",
			Age:       20,
		}, nil
	}
	return nil, nil
}

启动类的配置路径换成项目的引用路径

var configFile = flag.String("f", "quick_start/student_service/etc/student.yaml", "the config file")

点击启动

查看etcd keeper (etcd 的ui 插件) 可以看到已经成功注册了

在这里插入图片描述

提一下其中生成的studentservice 该目录包含了go-zero生成的接口交互服务文件,到时候服务之间交互就是引入的该文件


package studentservice

import (
	"context"

	"quickstart/quick_start/student_service/student"

	"github.com/zeromicro/go-zero/zrpc"
	"google.golang.org/grpc"
)

type (
	StudentRequest  = student.StudentRequest
	StudentResponse = student.StudentResponse

	StudentService interface {
		GetStudentInfo(ctx context.Context, in *StudentRequest, opts ...grpc.CallOption) (*StudentResponse, error)
	}
	defaultStudentService struct {
		cli zrpc.Client
	}
)

func NewStudentService(cli zrpc.Client) StudentService {
	return &defaultStudentService{
		cli: cli,
	}
}

func (m *defaultStudentService) GetStudentInfo(ctx context.Context, in *StudentRequest, opts ...grpc.CallOption) (*StudentResponse, error) {
	client := student.NewStudentServiceClient(m.cli.Conn())
	return client.GetStudentInfo(ctx, in, opts...)
}

编写order-service

最底部的学生服务被定义好后 ,那么就要编写订单服务,和学生服务不同的是,订单服务因为调用了学生服务,那么订单服务就要聚合学生服务

在order目录编写该文件

编写order.proto

syntax = "proto3";
option go_package="./order";
package order;

service OrderService {
rpc GetOrderInfo (OrderRequest) returns (OrderResponse);
}

message OrderRequest {
int32 order_id = 1;
}

message OrderResponse {
int32 order_id = 1;
int32 student_id = 2;
string data = 3;
}

该文件也是定义了一个接口,以及其中响应体定义的data 用于装载student服务响应的json对象字符串

goctl rpc protoc order.proto --go_out=. --go-grpc_out=. --zrpc_out=.

运行配置文件

自身也是微服务中的一环除去自身注册之外还需要定义一个需要引用到的学生rpc接口配置

etic/服务.yaml

Name: order-service
ListenOn: 127.0.0.1:9090
StudentRpcConf:
  Etcd:
    Hosts:
      - localhost:2379
    Key: student.rpc

Etcd:
  Hosts:
    - 127.0.0.1:2379
  Key: order.rpc

上下文配置文件

因为需要使用到其他模块,go-zero服务之前的交互是通过封装的 zrpc等文件 所以需要引入上下文

internal中的config.go文件

package config

import "github.com/zeromicro/go-zero/zrpc"

// Config is the configuration structure
type Config struct {
	zrpc.RpcServerConf
	StudentRpcConf zrpc.RpcClientConf
}

上下文件之中进行注册

svc目录下的上下文文件

package svc

import (
	"github.com/zeromicro/go-zero/zrpc"
	"quickstart/quick_start/order_service/internal/config"
//引入的就是student中 goctl生成的交互客户端文件
	"quickstart/quick_start/student_service/studentservice"
)

type ServiceContext struct {
	Config     config.Config
	StudentRpc studentservice.StudentService
}

func NewServiceContext(c config.Config) *ServiceContext {
	return &ServiceContext{
		Config: c,
		//从配置文件服务端
		StudentRpc: studentservice.NewStudentService(zrpc.MustNewClient(c.StudentRpcConf)),
	}
}

好了那么就可以进行服务的逻辑实现 对应的逻辑实现

func (l *GetOrderInfoLogic) GetOrderInfo(req *order.OrderRequest) (*order.OrderResponse, error) {
	fmt.Println("订单服务服务被调用")
	fmt.Println("订单ID:", req.OrderId)
	if req.OrderId == 1 {
		// TODO: 模拟联查除订单id查询除学生id
		studentResp, err := l.svcCtx.StudentRpc.GetStudentInfo(l.ctx, &student.StudentRequest{StudentId: 1})
		if err != nil {
			return nil, err
		}
		fmt.Println("学生信息:", studentResp)
		toString, _ := jsonx.MarshalToString(studentResp)
		return &order.OrderResponse{
			OrderId:   req.OrderId,
			StudentId: studentResp.StudentId,
			Data:      toString,
		}, nil
	}
	return nil, nil
}

到此位置 俩个微服务接口也就写完了,只需要编写网关 ,用户游览器给网关这个web服务发起请求,然后传递id 1就可以完成交互

和上面的一样修改启动类配置为引用地址启动

成功注册到etcd

在这里插入图片描述

web网关

编写web.api文件

syntax = "v1"

type Request {
	Name string `path:"name,options=you|me"`
}

type OderRequest {
	ID int `path:"id"`
}

type Response {
	Message string `json:"message"`
}

type OrderResponse {
	Message string `json:"message"`
	Data    string `json:"data"`
	Code    int    `json:"code"`
}

service web-api {
	@handler WebHandler
	get /from/:name (Request) returns (Response)

	@handler OrderHandler
	get /order/:id (OderRequest) returns (OrderResponse)
}


有俩个接口 其中一个没有用 文件是从官网案列粘贴的模板 所以不管

web网关负责的是和直接用户交互,所以到时候其他服务获取的数据就装在data字段即可

goctl api go -api web.api -dir .

生成代码后 ,同理开发者只要写逻辑相关代码即可

配置文件

定义调用服务的相关配置


Name: web-api
Host: 0.0.0.0
Port: 8888
Mode: dev
OrderRpcConf:
  Etcd:
    Hosts:
      - localhost:2379
    Key: order.rpc

上下文运行环境配置中引入

这里引入的字段名和配置文件中对应,zrpc源码根据这个进行赋值

import (
	"github.com/zeromicro/go-zero/rest"
	"github.com/zeromicro/go-zero/zrpc"
)

type Config struct {
	rest.RestConf
	//定义引入的rpc服务
	OrderRpcConf zrpc.RpcClientConf
}

rpc服务注册到上下文

	"github.com/zeromicro/go-zero/zrpc"
	"quickstart/quick_start/order_service/orderservice"
	"quickstart/quick_start/web/internal/config"
)

type ServiceContext struct {
	Config   config.Config
	OrderRpc orderservice.OrderService
}

func NewServiceContext(c config.Config) *ServiceContext {
	return &ServiceContext{
		Config:   c,
		OrderRpc: orderservice.NewOrderService(zrpc.MustNewClient(c.OrderRpcConf)),
	}
}

logic完成调用逻辑

func (l *OrderLogic) Order(req *types.OderRequest) (resp *types.OrderResponse, err error) {
	// todo: add your logic here and delete this line

	id := req.ID
	fmt.Printf("Orderid:%d", id)
	//新建一个rpc接口需要的参数
	o := new(order.OrderRequest)
	o.OrderId = int32(id)
	if info, err := l.svcCtx.OrderRpc.GetOrderInfo(l.ctx, o); nil != err {
		fmt.Printf("远程调用rpc失败")
	} else {
		resp := new(types.OrderResponse)
		fmt.Println("订单信息:", info)

		resp.Data = info.Data
		resp.Code = 0
		resp.Message = "服务调用成"
		return resp, nil
	}
	//t := new(types.Response)

	return
}

启动网关

api文件中定义的路由 ,是restful形式

get /order/:id (OderRequest) returns (OrderResponse)

生成的代码自然也是

func RegisterHandlers(server *rest.Server, serverCtx *svc.ServiceContext) {
	server.AddRoutes(
		[]rest.Route{
			{
				Method:  http.MethodGet,
				Path:    "/from/:name",
				Handler: WebHandler(serverCtx),
			},
			{
				Method:  http.MethodGet,
				Path:    "/order/:id",
				Handler: OrderHandler(serverCtx),
			},
		},
	)
}

游览器访问

http://localhost:8888/order/1

成功输出

在这里插入图片描述

被调用服务日志也是成功输出

在这里插入图片描述

那么go-zero微服务的快速开发模式其实以及了解完全了,
剩下的就是对框架本身的一些规则感觉和gin这些go web框架差不多的部分了,中间件,路由绑定,参数和返回和其他数据库,微服务的中间件的整合了,由于篇幅问题 下次笔者在进行书写

go-zero是一个开源的Go语言框架,它在构建微服务和高并发应用方面具有突破性的优势。其中一个突出的特点就是它整合了masterminds/squirrel,从而实现了优雅的多数据库支持。 masterminds/squirrel是一个流行的SQL查询构建器,它以非常直观和灵活的方式提供了编写SQL查询的功能。而go-zero在此基础上做了进一步的封装和优化,使得使用者能够更加方便地编写和执行SQL查询。 首先,go-zero提供了一组简洁而强大的API,使得构建SQL查询非常容易。开发者只需要按照一定的约定来创建查询参数和条件,然后使用go-zero提供的API来构建查询语句,即可完成复杂的SQL查询。 其次,go-zero还增加了一些高级功能,进一步提升了多数据库查询的灵活性和性能。例如,它支持数据库连接池管理,可以动态调整数据库连接数以适应并发请求;还支持分表分库功能,可以按照一定的规则将数据分散存储在不同的数据库或表中,从而提高查询效率。 最重要的是,go-zero通过内置的代码生成工具,提供了自动化生成数据库访问代码的能力。开发者只需要定义数据表的结构,然后运行代码生成工具,就能够自动生成包含增删改查等一系列数据库操作的代码。这极大地提高了开发效率,减少了出错的机会。 综上所述,go-zero整合了masterminds/squirrel,通过提供简洁强大的API、高级功能和自动化代码生成工具,实现了优雅的多数据库支持。它在微服务和高并发应用场景下的表现突出,为开发者提供了极大的便利和效率。
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包

打赏作者

蓝胖子不是胖子

你的鼓励将是我创作的最大动力

¥1 ¥2 ¥4 ¥6 ¥10 ¥20
扫码支付:¥1
获取中
扫码支付

您的余额不足,请更换扫码支付或充值

打赏作者

实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值