Golang 中的全局变量
Posted
技术标签:
【中文标题】Golang 中的全局变量【英文标题】:Global Vars in Golang 【发布时间】:2022-01-23 06:57:10 【问题描述】:我一直在阅读一本关于 Golang 的书,通常使用全局变量来公开私有函数的结果。为什么不公开公共功能?这是一个例子。 我来自java,这对我来说很棘手。使用这样的全局变量有什么好处?
编辑:我注意到没有括号我没有调用该函数。这只是一个指针。这是一个好的做法还是有一些优势?
package config
import "time"
var (
GetHTTPServerAddress = getHTTPServerAddress
GetHTTPReadTimeout = getHTTPReadTimeout
GetHTTPWriteTimeout = getHTTPWriteTimeout
)
func getHTTPServerAddress() string
return getConfigString("http.server_address")
func getHTTPReadTimeout() time.Duration
return getConfigDuration("http.read_timeout")
func getHTTPWriteTimeout() time.Duration
return getConfigDuration("http.write_timeout")
【问题讨论】:
请注意,您的代码不会公开调用函数的结果。公开结果将是GetHTTPServerAddress = getHTTPServerAddress()
,请注意括号。但如果你这样做,只需重命名函数以使其公开:func GetHTTPServerAddress()
在 GO 中它是“Exported”(以大写开头,可从其他包访问)或“Unexported”(以小写开头,只能从它定义的包内或在它自己的结构内访问) .导出的变量不是“全局”的,因为您必须导入包才能访问导出的变量/方法/等......并且在您的示例中,还不如导出函数(IE:Capitalize),您本质上是使用导出的变量作为存根以一种迂回的方式这样做
【参考方案1】:
我一直在读一本关于 Golang 的书,通常使用全局变量来公开私有函数的结果。
公开私有函数的结果,而不是函数本身;你的例子有点不对劲。如果没有括号,您将暴露函数本身。
package main
import (
"fmt"
)
var(
FooItself = foo // (func() string)(0x108b720)
FooResult = foo() // "bar"
)
func main()
fmt.Printf("%#v\n", FooItself)
fmt.Printf("%#v\n", FooResult)
func foo() string
return "bar"
Try it.
你的例子看起来更像这样。
package config
var (
// You wouldn't call the variable "Get*", that exposes
// the implementation.
HTTPServerAddress = getHTTPServerAddress()
HTTPReadTimeout = getHTTPReadTimeout()
HTTPWriteTimeout = getHTTPWriteTimeout()
)
func getHTTPServerAddress() string
return getConfigString("http.server_address")
func getHTTPReadTimeout() time.Duration
return getConfigDuration("http.read_timeout")
func getHTTPWriteTimeout() time.Duration
return getConfigDuration("http.write_timeout")
但不需要私有方法。只需运行该函数并将其分配给全局。
package config
var (
HTTPServerAddress = getConfigString("http.server_address")
HTTPReadTimeout = getConfigDuration("http.read_timeout")
HTTPWriteTimeout = getConfigDuration("http.write_timeout")
)
现在人们可以使用config.HTTPServerAddress
并且细节保持隐藏,不需要额外的一层方法。
是的,它可以在结果公共函数上完成。一个非常常见的例子是错误代码。例如,在net/http
。
var (
ErrBodyNotAllowed = errors.New("http: request method or response status code does not allow body")
ErrHijacked = errors.New("http: connection has been hijacked")
)
在 Java 中,您可以为每种类型的错误创建一个异常子类。在 Go 中,您创建一个对象,将其作为全局变量共享,并且所有内容都引用该对象。
目的是向用户隐藏详细信息。错误的确切措辞可以更改,但只要每个人都引用net.http.ErrBodyNotAllowed
,他们的代码仍然可以工作。
io/fs
有一个更复杂的示例,公开其私有方法的结果。
var (
ErrInvalid = errInvalid() // "invalid argument"
ErrPermission = errPermission() // "permission denied"
ErrExist = errExist() // "file already exists"
ErrNotExist = errNotExist() // "file does not exist"
ErrClosed = errClosed() // "file already closed"
)
func errInvalid() error return oserror.ErrInvalid
func errPermission() error return oserror.ErrPermission
func errExist() error return oserror.ErrExist
func errNotExist() error return oserror.ErrNotExist
func errClosed() error return oserror.ErrClosed
为什么在这里使用私有方法包装器?为什么不ErrInvalid = oserror.ErrInvalid
?
(我假设)Go 想要隐藏 oserror
,even from the documentation!
oserror
只是制作错误对象来定义其可能的错误。
package oserror
import "errors"
var (
ErrInvalid = errors.New("invalid argument")
ErrPermission = errors.New("permission denied")
ErrExist = errors.New("file already exists")
ErrNotExist = errors.New("file does not exist")
ErrClosed = errors.New("file already closed")
)
【讨论】:
我注意到没有括号我没有调用该函数。这只是一个指针。这是一个好的做法还是有一些优势?例如,我会调用 HTTPServerAddress() - 带括号,带括号的 var 和函数将被执行,因为没有括号的 getHTTPServerAddress 它只是函数 getHTTPServerAddress() 的指针。有意义吗? @lovelythyme 当您想要其他东西稍后调用该函数时,您可以使用函数指针。通常称为“处理程序”和“回调”。例如,registering handlers for HTTP routes。http.HandleFunc("/foo", handleFoo)
表示当请求 /foo
时调用 handleFoo
函数。或traverse a directory tree and call a function on each file。 Java can do the same thing,我相信它叫Runnable
。【参考方案2】:
如果是函数的结果,可能类似于static
成员初始化,甚至可能是static final
。
您链接的示例表明它是某种配置加载过程,并且值公开以供全局使用。
预计算数据也并非闻所未闻。
但是,如果您看到函数指针,则可以启用模拟。
YMMV。有时它也只是一种反模式。
【讨论】:
以上是关于Golang 中的全局变量的主要内容,如果未能解决你的问题,请参考以下文章
Golang✔️走进 Go 语言✔️ 第九课 局部变量 vs 全局变量