分享
  1. 首页
  2. 文章

go-micro 框架源码剖析 之 函数选项模式

Chole121 · · 1074 次点击 · · 开始浏览
这是一个创建于 的文章,其中的信息可能已经有所发展或是发生改变。

近期在查阅go-micro源码过程中发现,在设置创建微服务的参数选项时都依赖于项目外层一个名为github.com/micro/go-micro/options.go的文件,它定义了创建一个微服务所需要的各种参数选项,其实现过程中使用的方法值得借鉴。

1、创建微服务示例

在go-micro中使用micro.NewService创建一个微服务

import "github.com/micro/go-micro"
service := micro.NewService()

也可以在创建过程中设置服务选项(如服务名称,自定义服务发现等)

service := micro.NewService(
 micro.Name("hellooo"),
 micro.Version("latest"),
)

对比两个例子可以发现,虽然go语言不支持默认参数,但是上面的代码例子却达到了同样的效果。

2、如何实现默认参数

那么,go-micro是如何实现默认参数的呢?进入micro.NewService源码看个究竟:
github.com/micro/go-micro/go-micro.go

// Option定义为一个方法,接受Options类型的指针参数
// 注意这个Option类型(函数类型),这是理解本文的关键
type Option func(*Options)
...
// 创建并返回一个服务:接收Option函数类型的不定向参数列表
func NewService(opts ...Option) Service {
 return newService(opts...)
}
...
// 内部真正创建服务的方法
func newService(opts ...Option) Service {
 // 关键步骤:利用opts函数列表初始化服务
 options := newOptions(opts...)
 ...
 // 返回初始化的服务
 return &service{
 opts: options,
 }
}

其中newOptions方法最为关键,继续跟踪下去:
github.com/micro/go-micro/options.go,该文件定义了设置服务相关的所有选项:

// 服务选项结构体
// 因为go-micro是可插件化的框架,其中的组件(如消息中间件)均是可以用其他类似服务替换的
type Options struct {
 Broker broker.Broker 
 Cmd cmd.Cmd
 Client client.Client
 Server server.Server
 Registry registry.Registry
 Transport transport.Transport
 // Register loop interval
 RegisterInterval time.Duration
 // Before and After funcs
 BeforeStart []func() error
 BeforeStop []func() error
 AfterStart []func() error
 AfterStop []func() error
 Context context.Context
}
// 生成服务相关选项(初始化使用go-micro框架定义的默认组件)
func newOptions(opts ...Option) Options {
 // 初始化默认值
 opt := Options{
 Broker: broker.DefaultBroker,
 Cmd: cmd.DefaultCmd,
 Client: client.DefaultClient,
 Server: server.DefaultServer,
 Registry: registry.DefaultRegistry,
 Transport: transport.DefaultTransport,
 Context: context.Background(),
 }
 for _, o := range opts {
 o(&opt) // 依次调用opts函数列表中的函数,为服务选项(opt变量)赋值
 }
 return opt
}
...
// 服务名字选项
// 返回一个Option类型的函数(闭包):接受Options类型指针参数并修改之
func Name(n string) Option {
 return func(o *Options) {
 o.Server.Init(server.Name(n))
 }
}
// 服务版本选项
// 返回一个Option类型的函数(闭包):接受Options类型指针参数并修改之
func Version(v string) Option {
 return func(o *Options) {
 o.Server.Init(server.Version(v))
 }
}
...

除了Name与Version选项之外,go-micro的Broker、Register、Transport等选项均是通过此方法实现的。它极大地利用了go语言支持闭包的特性,优雅地实现了函数支持默认参数
的功能。
再次回到文章开头出创建服务的代码就很好理解了:

// 创建服务,同时接收对指定参数选项的设置
service := micro.NewService(
 micro.Name("hellooo"),
 micro.Version("latest"),
)

这样的写法处理调用代码比较简洁之外,它扩展性特别好,增加新的选项时,只需要很少量的代码。

这便是go语言的函数选项模式。

3、函数选项(Functional Options)模式

函数选项模式是由Rob Pike提出,并由Dave Cheney等推广开,它优雅地解决了go语言中默认参数问题。

虽然Functional Options并不是一个新的概念,从Rob Pike提出至今已4年有余(2014),但作为Golang新手,还是很受启发。这也是阅读源码带来的好处。

下面总结一下,函数选项模式有哪些优点:

  • 支持默认参数:不必向结构体参数那样,不使用时仍必须参数一个空的struct值
  • 代码简洁:即使是像go-micro这种支持如此繁多选项,代码也很美观
  • 扩展性好:增加新的选项只需少量代码

推而广之:类似结构体中变量的赋值都可以效仿之。

4、 ReFerences

https://commandcenter.blogspo...
https://dave.cheney.net/2014/...
https://halls-of-valhalla.org...
https://lingchao.xin/post/fun...

本文来自:Segmentfault
感谢作者:yabohe
查看原文:go-micro 框架源码剖析 之 函数选项模式

添加小编微信:grey0805,加入知识学习小分队~!


有疑问加站长微信联系(非本文作者)

本文来自:简书

感谢作者:Chole121

查看原文:go-micro 框架源码剖析 之 函数选项模式

入群交流(和以上内容无关):加入Go大咖交流群,或添加微信:liuxiaoyan-s 备注:入群;或加QQ群:692541889

关注微信
1074 次点击
暂无回复
添加一条新回复 (您需要 后才能回复 没有账号 ?)
  • 请尽量让自己的回复能够对别人有帮助
  • 支持 Markdown 格式, **粗体**、~~删除线~~、`单行代码`
  • 支持 @ 本站用户;支持表情(输入 : 提示),见 Emoji cheat sheet
  • 图片支持拖拽、截图粘贴等方式上传

用户登录

没有账号?注册
(追記) (追記ここまで)

今日阅读排行

    加载中
(追記) (追記ここまで)

一周阅读排行

    加载中

关注我

  • 扫码关注领全套学习资料 关注微信公众号
  • 加入 QQ 群:
    • 192706294(已满)
    • 731990104(已满)
    • 798786647(已满)
    • 729884609(已满)
    • 977810755(已满)
    • 815126783(已满)
    • 812540095(已满)
    • 1006366459(已满)
    • 692541889

  • 关注微信公众号
  • 加入微信群:liuxiaoyan-s,备注入群
  • 也欢迎加入知识星球 Go粉丝们(免费)

给该专栏投稿 写篇新文章

每篇文章有总共有 5 次投稿机会

收入到我管理的专栏 新建专栏