一、Http框架Gin
推荐指数:⭐⭐⭐⭐⭐
Gin是一个用Go(Golang)编写得HTTP web框架,拥有更好性能得API框架,文档非常完全,很多微服务也很方便集成。Gin绝对优秀~
touch example.go
package main
import "github.com/gin-gonic/gin"
func main() {
r := gin.Default()
r.GET("/ping", func(c *gin.Context) {
c.JSON(200, gin.H{
"message": "pong",
})
})
r.Run() // 监听并在 0.0.0.0:8080 上启动服务
}
go run example.go
二、CLI 命令(spf13/cobra)
推荐指数:⭐⭐⭐⭐⭐
Cobra既是一个创建强大的现代CLI应用程序的库,也是一个生成应用程序和命令的程序。
可以使用这个库来管理命令应用程序,执行runner应用程序,初始化配置,病启动Reast API。
基于Cobra的应用组织结构:
├── app
│ ├── main.go
│ ├── cmd
│ └── root.go
在 app/main.go 中:
package main
import (
"app/cmd"
)
func main() {
cmd.Execute()
}
在 app/cmd/root.go 中:
package cmd
var rootCmd = &cobra.Command{
Use: "hugo",
Short: "Hugo is a very fast static site generator",
Long: `A Fast and Flexible Static Site Generator built with love by spf13 and friends in Go. Complete documentation is available at http://hugo.spf13.com`,
Run: func(cmd *cobra.Command, args []string) {
// Do Stuff Here
},
}
func Execute() {
if err := rootCmd.Execute(); err != nil {
fmt.Println(err)
os.Exit(1)
}
}
三、ORM:Gorm
推荐指数:⭐⭐⭐⭐
推荐理由:代码优雅,场景非常丰富,满足我们的SQL场景需求,比如:
- 全功能ORM
- 关联(拥有一个、拥有多个、属于、多对多、多态、单表继承)
- Create,Save,Update,Delete,Find 中钩子方法
- 支持 Preload、Joins 的预加载
- 事务,嵌套事务,Save Point,Rollback To to Saved Point
- Context、预编译模式、DryRun 模式
- 批量插入,FindInBatches,Find/Create with Map,使用 SQL 表达式、Context Valuer 进行 CRUD
- SQL 构建器,Upsert,锁,Optimizer/Index/Comment Hint,命名参数,子查询
- 复合主键,索引,约束
- 自动迁移
- 自定义 Logger
- 灵活的可扩展插件 API:Database Resolver(多数据库,读写分离)、Prometheus…
- 每个特性都经过了测试的重重考验
四、SQL(jmoiron/sqlx)
推荐指数:⭐⭐⭐⭐
是sqlx的一个库,在Go的标准database/sql库上提供了一组扩展。
有一点很舒服,是可以可以进行结构扫码,快而简单!
结构扫描示例:
place := Place{}
rows, err := db.Queryx("SELECT * FROM place")
for rows.Next() {
err := rows.StructScan(&place)
if err != nil {
log.Fatalln(err)
}
fmt.Printf("%#v\n", place)
}
五、配置管理:viper
推荐指数:⭐⭐⭐⭐
- 支持JSON/TOML/YAML/HC:/en:L/envfile/Java properties等多种格式的配置文件;
- 可以设置监听配置文件的修改,修改时自动加载新的配置;
- 从环境变量、命令行选项和io。Reader中读取配置;
- 从远程配置系统中读取和监听修改,如etcd/Consu;
- 代码逻辑中显示设置键值
六、日志管理:zap
推荐指数:⭐⭐⭐⭐
logger, _ := zap.NewProduction()
defer logger.Sync() // flushes buffer, if any
sugar := logger.Sugar()
sugar.Infow("failed to fetch URL",
// Structured context as loosely typed key-value pairs.
"url", url,
"attempt", 3,
"backoff", time.Second,
)
sugar.Infof("Failed to fetch URL: %s", url)
七、Web框架(labstack/echo)
推荐指数:⭐⭐⭐⭐
高性能、极简主义的Go Web框架。
安装:
// go get github.com/labstack/echo/{version}
go get github.com/labstack/echo/v4
示例:
package main
import (
"net/http"
"github.com/labstack/echo/v4"
"github.com/labstack/echo/v4/middleware"
)
func main() {
// Echo instance
e := echo.New()
// Middleware
e.Use(middleware.Logger())
e.Use(middleware.Recover())
// Routes
e.GET("/", hello)
// Start server
e.Logger.Fatal(e.Start(":1323"))
}
// Handler
func hello(c echo.Context) error {
return c.String(http.StatusOK, "Hello, World!")
}
八、Redis
推荐指数:⭐⭐⭐⭐
如果你用的缓存或队列是Redis,那就用它!
九、消息队列 asynq
推荐指数:⭐⭐⭐⭐
可靠、简单、高效的分布式任务队列。
运行实例:
十、消息传递 NSQ
推荐指数:⭐⭐⭐⭐
NSQ 拓扑
NSQ 组件:
- nsqlookupd (守护进程管理拓扑 / 路由)
- nsqd(守护进程管理接收、排队和传递消息)
- nsqadmin(nsq 的默认 Web UI)
docker-compose 示例:(nsqlookupd, nsqd, nsqadmin)
version: '3'
services:
nsqlookupd:
image: nsqio/nsq
command: /nsqlookupd
ports:
- "4160:4160"
- "4161:4161"
nsqd:
image: nsqio/nsq
command: /nsqd --lookupd-tcp-address=nsqlookupd:4160
depends_on:
- nsqlookupd
ports:
- "4150:4150"
- "4151:4151"
nsqadmin:
image: nsqio/nsq
command: /nsqadmin --lookupd-http-address=nsqlookupd:4161
depends_on:
- nsqlookupd
ports:
- "4171:4171"
执行:
运行 docker:
$ docker-compose up -d
或者,如果使用名称 (docker-compose-nsq.yml):
$ docker-compose -f docker-compose-nsq.yml up -d
检查容器 docker:
$ docker-compose ps
查看日志:
$ docker-compose logs
检查 nsq Web UI(假设端口为 32770):
$ curl http://127.0.0.1:32770/ping
在golang中:
创建文件夹:
├── consume
│ └── consume.go
└── publish
└── publish.go
consume.go:
package main
import (
"log"
"sync"
"github.com/nsqio/go-nsq"
)
func main() {
wg := &sync.WaitGroup{}
wg.Add(1)
decodeConfig := nsq.NewConfig()
c, err := nsq.NewConsumer("My_NSQ_Topic", "My_NSQ_Channel", decodeConfig)
if err != nil {
log.Panic("Could not create consumer")
}
c.AddHandler(nsq.HandlerFunc(func(message *nsq.Message) error {
log.Println("NSQ message received:")
log.Println(string(message.Body))
return nil
}))
err = c.ConnectToNSQD("127.0.0.1:4150")
if err != nil {
log.Panic("Could not connect")
}
log.Println("Awaiting messages from NSQ topic \"My NSQ Topic\"...")
wg.Wait()
}
运行 consume.go:
$ go run consume/consume.go
publish.go:
package main
import (
"log"
"github.com/nsqio/go-nsq"
)
func main() {
config := nsq.NewConfig()
p, err := nsq.NewProducer("127.0.0.1:4150", config)
if err != nil {
log.Panic(err)
}
err = p.Publish("My_NSQ_Topic", []byte("sample NSQ message"))
if err != nil {
log.Panic(err)
}
运行 publish.go:
$ go run publish/publish.go