https://go-zero.dev/cn/docs/prepare/golang-install
成都創(chuàng)新互聯(lián)公司從2013年開始,是專業(yè)互聯(lián)網(wǎng)技術(shù)服務(wù)公司,擁有項(xiàng)目做網(wǎng)站、網(wǎng)站設(shè)計(jì)網(wǎng)站策劃,項(xiàng)目實(shí)施與項(xiàng)目整合能力。我們以讓每一個(gè)夢(mèng)想脫穎而出為使命,1280元西華做網(wǎng)站,已為上家服務(wù),為西華各地企業(yè)和個(gè)人服務(wù),聯(lián)系電話:18982081108
$ go version
go version go1.15.1 darwin/amd64
$ tar -C /usr/local -xzf go1.15.8.linux-amd64.tar.gz
$ $HOME/.profile
$ export PATH=$PATH:/usr/local/go/bin
$ source $HOME/.profile
$ go version
go version go1.15.1 linux/amd64
$ go version
go version go1.15.1 windows/amd64
Go Module是Golang管理依賴性的方式,像Java中的Maven,Android中的Gradle類似。
$ go env GO111MODULE
on
$ go env -w GO111MODULE="on"
$ go env -w GOPROXY=https://goproxy.cn
查看GOMODCACHE
$ go env GOMODCACHE
go env -w GOMODCACHE=$GOPATH/pkg/mod
Goctl在go-zero項(xiàng)目開發(fā)著有著很大的作用,其可以有效的幫助開發(fā)者大大提高開發(fā)效率,減少代碼的出錯(cuò)率,縮短業(yè)務(wù)開發(fā)的工作量,更多的Goctl的介紹請(qǐng)閱讀Goctl介紹
### 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
go install github.com/zeromicro/go-zero/tools/goctl@latest
sudo vim /etc/paths //添加環(huán)境變量
在最后一行添加如下內(nèi)容 //$GOPATH 為你本機(jī)上的文件地址
$GOPATH/bin
$ goctl -v
goctl version 1.1.4 darwin/amd64
goctl api new greet
cd greet
go mod init
go mod tidy
go run greet.go -f etc/greet-api.yaml
greet-api.yaml
配置文件里修改,此時(shí),可以通過(guò) curl 請(qǐng)求,或者直接在瀏覽器中打開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-ebcf3e308-00
Date: Thu, 22 Oct 2020 14:03:18 GMT
Content-Length: 14
null
$ 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
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) //此方法是注冊(cè)路由和路由映射Handler,重點(diǎn)在這里
fmt.Printf("Starting server at %s:%d...\n", c.Host, c.Port)
server.Start()
}
internal\handler\routes.go
中func RegisterHandlers(server *rest.Server, serverCtx *svc.ServiceContext) {
server.AddRoutes( //往rest.Server中添加路由
[]rest.Route{ //路由數(shù)組
{
Method: http.MethodGet,
Path: "/from/:name", //路由
Handler: GreetHandler(serverCtx),//當(dāng)前路由的處理Handler
},
},
)
}
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 { //請(qǐng)求的錯(cuò)誤判斷,這個(gè)可以不用管
3. httpx.Error(w, err)
4. return
5. }
l := logic.NewGreetLogic(r.Context(), ctx) //GreetHandler處理函數(shù)將請(qǐng)求轉(zhuǎn)發(fā)到了GreetLogic中,調(diào)用NewGreetLogic進(jìn)行結(jié)構(gòu)體的初始化
resp, err := l.Greet(req) //然后調(diào)用Greet來(lái)進(jìn)行處理請(qǐng)求,所以我們?cè)贕reetLogic.Greet方法中可以看到一句話// todo: add your logic here and delete this line
if err != nil {
httpx.Error(w, err)
} else {
httpx.OkJson(w, resp)
}
}
}
在路由注冊(cè)時(shí),我們?nèi)绻?wù)越加越多,那么相對(duì)應(yīng)的func xxxxHandler(ctx *svc.ServiceContext) http.HandlerFunc
就要進(jìn)行多次的添加,并且這個(gè)方法體內(nèi)部1到5行是屬于額外的重復(fù)添加
例如:我們添加一個(gè)customlogic.go
按照命名的正確和規(guī)范性,需要在internal\logic
目錄下添加customlogic.go文件,然后在internal\handler
目錄下添加customhandler.go文件,并且兩個(gè)文件都添加相對(duì)應(yīng)的結(jié)構(gòu)體和函數(shù)等,最后在routes.go
中再添加一次
{
Method: http.MethodGet,
Path: "/custom/:name",
Handler: CustomHandler(serverCtx),
},
此時(shí),我們的文件結(jié)構(gòu)應(yīng)該是這樣
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
當(dāng)單體應(yīng)用達(dá)到一定的數(shù)量級(jí),handler和logic文件夾下將會(huì)同步增加很多的文件
自Go1.18開始,go開始使用泛型,泛型的廣泛定義 :是一種把明確類型的工作推遲到創(chuàng)建對(duì)象或者調(diào)用方法的時(shí)候才去明確的特殊的類型。 也就是說(shuō)在泛型使用過(guò)程中,操作的數(shù)據(jù)類型被指定為一個(gè)參數(shù),而這種參數(shù)類型可以用在 類、方法和接口 中,分別被稱為 泛型類 、 泛型方法 、 泛型接口 。
我們可以利用泛型,讓在添加路由時(shí)就要固定死的Handler: GreetHandler(serverCtx)
推遲到后面,去根據(jù)實(shí)際的Logic結(jié)構(gòu)體去判斷需要真正執(zhí)行的logic.NewGreetLogic(r.Context(), ctx)
初始化結(jié)構(gòu)體和l.Greet(req)
邏輯處理方法
internal\logic
下添加一個(gè)baselogic.go
文件,參考Go泛型實(shí)戰(zhàn) | 如何在結(jié)構(gòu)體中使用泛型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) //每一個(gè)結(jié)構(gòu)體中必須要繼承一下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) { //作為一個(gè)中轉(zhuǎn)處理方法,最終執(zhí)行結(jié)構(gòu)體的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
}
//通過(guò)泛型動(dòng)態(tài)調(diào)用不同結(jié)構(gòu)體的Handler方法
cc := logic.New[T]()
cc.LogicHandler(req, w, r, svcCtx)
}
}
internal\logic\greetlogic.go
中增加一個(gè)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{ //路由數(shù)組
{
Method: http.MethodGet,
Path: "/from/:name", //路由
Handler: BaseHandlerFunc(serverCtx,logic.GreetLogic{}),
},
},
)
}
現(xiàn)在就大功告成了,我們啟動(dòng)一下
go run greet.go -f etc/greet-api.yaml
然后在瀏覽器中請(qǐng)求一下http://localhost:8888/from/you
internal\logic
下新增一個(gè)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稍微修改了一下,便于區(qū)分
// 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{ //路由數(shù)組
{
Method: http.MethodGet,
Path: "/from/:name", //路由
Handler: BaseHandlerFunc(serverCtx,logic.GreetLogic{}),
},
{
Method: http.MethodGet,
Path: "/to/:name", //路由
Handler: BaseHandlerFunc(serverCtx,logic.CustomLogic{}),
},
},
)
}
其他地方不需要修改
我們啟動(dòng)一下
go run greet.go -f etc/greet-api.yaml
然后在瀏覽器中請(qǐng)求一下http://localhost:8888/from/you
、http://localhost:8888/to/you
、http://localhost:8888/too/you
現(xiàn)在,在添加新的logic做路由映射時(shí),就可以直接簡(jiǎn)化掉添加xxxxhandler.go
文件了,實(shí)際上是將這個(gè)Handler移動(dòng)到了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)
//通過(guò)泛型動(dòng)態(tài)調(diào)用不同結(jié)構(gòu)體的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) //每一個(gè)結(jié)構(gòu)體中必須要繼承一下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)
// }
這樣就可以了