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 全局变量

Golang 模板 - 使用模板变量作为范围循环内的全局变量

golang的flag包源码解析与使用

Golang基本语法

golang 变量

Golang:变量的声明