不知不觉写 Go
已经快一年了,上线了大大小小好几个项目;心态也经历了几轮变化。
因为我个人大概前五年时间写的是 Java
,中途写过一年多的 Python
,所以刚接触到 Go 时的感觉如下图:
既没有 Java
的生态,也没有 Python
这么多语法糖。
写到现在的感觉就是:
这里就不讨论这几门语言谁强谁弱了;重点和大家分享下我们日常开发中所使用到的一些第三方库与工具。
这里我主要将这些库分为两类:
首先是业务开发,主要包含了 web
、数据库、Redis
等。
首先是 Gin,一款 HTTP 框架,使用简单、性能优秀、资料众多;你还在犹豫选择哪款框架时,那就选择它吧,基本没错。
当然和它配套的 github.com/swaggo/gin-swagger swagger 工具也是刚需;利用它可以生成 swagger 文档。
GORM 也没啥好说的,如果你喜欢 orm
的方式操作数据库,那就选它吧;同样的也是使用简单、资料较多。
如果有读写分离需求,也可以使用 GORM
官方提供的插件 https://github.com/go-gorm/dbresolver ,配合 GORM
使用也是非常简单。
Go 语言自身提供的错误处理比较简单,https://github.com/pkg/errors 提供了更强大的功能,比如:
常用的有以下 API:
// WithMessagef annotates err with the format specifier.
func WithMessagef(err error, format string, args ...interface{}) error
// WithStack annotates err with a stack trace at the point WithStack was called.
func WithStack(err error) error
Go 里的日志打印库非常多,日志在日常开发中最好就是存在感低;也就是说性能强(不能影响到业务代码)、使用 API 简单。
"github.com/rs/zerolog/log"
log.Debug().Msgf("OrderID :%s", "12121")
https://github.com/qax-os/excelize是一个读写 Excel 的库,基本上你能遇到的 Excel 操作它都能实现。
https://github.com/jinzhu/now 是一个时间工具库:
获取当前的年月日、时分秒。
不同时区支持。
最后一周、最后一个月等。
import "github.com/jinzhu/now"
time.Now() // 2013-11-18 17:51:49.123456789 Mon
now.BeginningOfMinute() // 2013-11-18 17:51:00 Mon
now.BeginningOfHour() // 2013-11-18 17:00:00 Mon
now.BeginningOfDay() // 2013-11-18 00:00:00 Mon
now.BeginningOfWeek() // 2013-11-17 00:00:00 Sun
now.BeginningOfMonth() // 2013-11-01 00:00:00 Fri
now.BeginningOfQuarter() // 2013-10-01 00:00:00 Tue
now.BeginningOfYear() // 2013-01-01 00:00:00 Tue
now.EndOfMinute() // 2013-11-18 17:51:59.999999999 Mon
now.EndOfHour() // 2013-11-18 17:59:59.999999999 Mon
now.EndOfDay() // 2013-11-18 23:59:59.999999999 Mon
now.EndOfWeek() // 2013-11-23 23:59:59.999999999 Sat
now.EndOfMonth() // 2013-11-30 23:59:59.999999999 Sat
now.EndOfQuarter() // 2013-12-31 23:59:59.999999999 Tue
now.EndOfYear() // 2013-12-31 23:59:59.999999999 Tue
now.WeekStartDay = time.Monday // Set Monday as first day, default is Sunday
now.EndOfWeek() // 2013-11-24 23:59:59.999999999 Sun
当业务上需要精度计算时 https://github.com/shopspring/decimal 可以帮忙。
import (
"fmt"
"github.com/shopspring/decimal"
)
func main() {
price, err := decimal.NewFromString("136.02")
quantity := decimal.NewFromInt(3)
fee, _ := decimal.NewFromString(".035")
taxRate, _ := decimal.NewFromString(".08875")
subtotal := price.Mul(quantity)
preTax := subtotal.Mul(fee.Add(decimal.NewFromFloat(1)))
total := preTax.Mul(taxRate.Add(decimal.NewFromFloat(1)))
fmt.Println("Subtotal:", subtotal) // Subtotal: 408.06
fmt.Println("Pre-tax:", preTax) // Pre-tax: 422.3421
fmt.Println("Taxes:", total.Sub(preTax)) // Taxes: 37.482861375
fmt.Println("Total:", total) // Total: 459.824961375
fmt.Println("Tax rate:", total.Sub(preTax).Div(preTax)) // Tax rate: 0.08875
}
基本上你能想到的精度转换它都能做到;配合上 GORM
也可以将 model
字段声明为 decimal
的类型,数据库对应的也是 decimal
,这样使用起来时会更方便。
Amount decimal.Decimal `gorm:"column:amout;default:0.0000;NOT NULL" json:"amout"`
https://github.com/jinzhu/configor 是一个配置文件读取库,支持 YAML/JSON/TOML
等格式。
https://github.com/patrickmn/go-cache 是一个类似于 Java 中的 Guava cache
,线程安全,使用简单;不需要分布式缓存的简单场景可以考虑。
c := cache.New(5*time.Minute, 10*time.Minute)
// Set the value of the key "foo" to "bar", with the default expiration time
c.Set("foo", "bar", cache.DefaultExpiration)
https://github.com/jinzhu/copier 看名字就知道这是一个数据复制的库,与 Java
中的 BeanUtils.copy()
类似;可以将两个字段相同但对象不同的 struct
进行数据复制,也支持深拷贝。
func Copy(toValue interface{}, fromValue interface{}) (err error)
在我们需要一个临时 struct 来存放数据时很有用,特别是一个 struct 中字段非常多时,一个个来回赋值确实有点费手指。
但也要注意不要什么情况都使用,会带来一些弊端:
总之在业务开发时,还是建议人工编写,毕竟代码是给人看的。
https://github.com/caarlos0/env 这个库可以将我们的环境变量转换为一个 struct
.
type config struct {
Home string `env:"HOME"`
}
func main() {
cfg := config{}
if err := env.Parse(&cfg); err != nil {
fmt.Printf("%+v\n", err)
}
fmt.Printf("%+v\n", cfg)
}
这个在我们打包代码到不同的运行环境时非常有用,利用它可以方便的获取不同环境变量。
https://github.com/mssola/user_agent 是一个格式化 user-agent
的小工具。
当我们需要在服务端收集 user-agen
时可以更快的读取数据。
func main() {
ua := user_agent.New("Mozilla/5.0 (Linux; U; Android 2.3.7; en-us; Nexus One Build/FRF91) AppleWebKit/533.1 (KHTML, like Gecko) Version/4.0 Mobile Safari/533.1")
fmt.Printf("%v\n", ua.Mobile()) // => true
fmt.Printf("%v\n", ua.Bot()) // => false
fmt.Printf("%v\n", ua.Mozilla()) // => "5.0"
fmt.Printf("%v\n", ua.Model()) // => "Nexus One"
fmt.Printf("%v\n", ua.Platform()) // => "Linux"
fmt.Printf("%v\n", ua.OS())
}
https://github.com/nyaruka/phonenumbers 手机号码验证库,可以不用自己写正则表达式了。
// parse our phone number
num, err := phonenumbers.Parse("6502530000", "US")
接下来是一些基础工具库,包含一些主流的存储的客户端、中间件等。
github.com/agiledragon/gomonkey 是一个 mock
打桩工具,当我们写单元测试时,需要对一些非接口函数进行 mock
会比较困难,这时就需要用到它了。
由于它是修改了调用对应函数时机器跳转指令,而 CPU 架构的不同对应的指令也不同,所以在我们使用时还不兼容苹果的 M1 芯片,不过目前应该已经兼容了,大家可以试试。
https://github.com/smartystreets/goconvey 也是配合单元测试的库,可以兼容 go test
命令。
提供可视化 web UI。
与 IDE 集成显示单元覆盖率。
https://github.com/uber-go/dig 这是一个依赖注入库,我们这里暂不讨论是否应该使用依赖注入,至少目前我们使用下来还是有几个好处:
当然也有一些不太方便的地方:
我们内部有自己开发一个业务框架,其中所有的对象都交由 dig 进行管理,使用起来倒也是比较方便。
https://github.com/spf13/cobra是一个功能强大的命令行工具库,我们用它来实现内部的命令行工具,同时也推荐使用 https://github.com/urfave/cli/ 我个人会更习惯用后者,要简洁一些。
https://github.com/uw-labs/bloomrpc 一个 gRPC
可视化工具,比起自己写 gRPC
客户端的代码那确实是要简单许多。
但也有些小问题,比如精度。如果是 int64 超过了 2^56 服务端拿到的值会发生错误,这点目前还未解决。
https://github.com/go-redis/redis/ Redis 客户端,没有太多可说的;发展了许多年,该有的的功能都有了。
https://github.com/olivere/elastic 这也是一个非常成熟的 elasticsearch
库。
https://github.com/go-resty/resty/ 一个 http client, 使用起来非常简单:
// Create a Resty Client
client := resty.New()
resp, err := client.R().
EnableTrace().
Get("https://httpbin.org/get")
有点 Python requests
包那味了。
Pulsar 官方出品的 go 语言客户端,相对于 Java 来说其他语言的客户端几乎都是后娘养的;功能会比较少,同时更新也没那么积极;但却没得选。
https://github.com/grpc-ecosystem/go-grpc-middleware 官方提供的 gRPC
中间件,可以自己实现内部的一些鉴权、元数据、日志等功能。
https://github.com/pilosa/go-pilosa 是一个位图数据库的客户端,位图数据库的场景应用比较有限,通常是有标签需求时才会用到;比如求 N 个标签的交并补集;数据有一定规模后运营一定会提相关需求;可以备着以备不时之需。
https://github.com/cheggaaa/pb 一个命令行工具进度条,编写命令行工具时使用它交互会更优雅。
最后我汇总了一个表格,方便查看:
名称
类型
功能
星级
Gin
业务开发
HTTP
框架
️️️️️
GORM
业务开发
ORM
框架
️️️️️
errors
业务开发
异常处理库
️️️️️
zorolog
业务开发
日志库
️️️️️
excelize
业务开发
Excel
相关需求
️️️️️
now
业务开发
时间处理
️️️️️
Decimal
业务开发
精度处理
️️️️️
configor
业务开发
配置文件
️️️️️
go-cache
业务开发
本地缓存
️️️
copier
业务开发
数据复制
️️️️️
env
业务开发
环境变量
️️️️️
user_agent
业务开发
读取 user-agent
️️️️️
phonenumbers
业务开发
手机号码验证
️️️️️
gomonkey
基础工具
mock
工具
️️️️
goconvey
基础工具
单测覆盖率
️️️️
dig
基础工具
依赖注入
️️️️
cobra
基础工具
命令行工具
️️️
cli
基础工具
命令行工具
️️️
BloomRPC
基础工具
gRPC
调试客户端
️️️
redis
基础工具
Redis 客户端
️️️
elastic
基础工具
elasticsearch
客户端
️️️
resty
基础工具
http 客户端
️️️
pulsar-client-go
基础工具
Pulsar
客户端
️️️
go-grpc-middleware
基础工具
gRPC
中间件
️️
go-pilosa
基础工具
pilosa
客户端
️️️
基础工具
命令行工具进度条
️️️
星级评分的规则主要是看实际使用的频次。
最后夹带一点私货(其实也谈不上)
文中提到了我们内部有基于以上库整合了一个业务开发框架;也基于该框架上线了大大小小10几个项目,改进空间依然不少,目前还是在快速迭代中。
大概的用法,入口 main.go
:
最后截取我在内部的分享就概括了整体的思想--引用自公司一司姓同事
。
也许我们内部经过多次迭代,觉得有能力开放出来给社区带来一些帮助时也会尝试开源;现阶段就不嫌丑了。
这些库都是我们日常开发最常用到的,也欢迎大家在评论区留下你们常用的库与工具。
手机扫一扫
移动阅读更方便
你可能感兴趣的文章