2

嗨,我正在尝试使我的功能单元可测试。其中一个建议是将函数分配给变量并使其全局可访问。我只是这样做了,但现在我遇到了一个初始化循环,下面是我的代码

////////////////////////
// main.go
////////////////////////

func DownloadFile(filename string) {
    // Initialized request variable here
    // ... doing initialization
    // End of initialization

    res = ProcessDownload(request)

    if res == 401 {
       return doRetry(filename)
    }

    // some code here
    return downloadFile(filename)
}

func DoRetry(filename string) {
    // Doing some database insert/updating
   return downloadFile(string)
}

在我的另一个文件中,我将此函数分配给变量

//////////////////////
// global.go
//////////////////////

var downloadFile = DownloadFile
var doRetry = DoRetry

我在 global.go 中这样做的原因是为了使DoRetryDownloadFile单元可测试。使用这种方法意味着我可以在不创建接口的情况下模拟该功能。因为这只是一个独立的功能,不需要在某个类中。变量的所有其余部分都可以,但是当涉及执行递归行为的函数时,我收到以下错误

./global.go:22:5: initialization loop:
/go/src/project/global.go:22:5 downloadFile refers to
/go/src/project/process.go:403:99 DownloadFile refers to
/go/src/project/global.go:23:5 doRetry refers to
/go/src/project/process.go:394:89 DoRetry refers to
/go/src/project/global.go:22:5 downloadFile
FAIL    project [build failed]

我知道这可能在这里重复,但我看不出这里的解决方案如何解决我的问题。

请帮助我。谢谢,我在此链接中阅读了有关模拟函数的信息

Go 中的模拟函数

https://husobee.github.io/golang/testing/unit-test/2015/06/08/golang-unit-testing.html

4

1 回答 1

5

解决方案是在声明downloadFiledoRetry函数变量时不赋值,而是将初始化“推迟”到包init()函数,如下所示:

var downloadFile func(string) //= DownloadFile
var doRetry func(string)      //= DoRetry

func init() {
    downloadFile = DownloadFile
    doRetry = DoRetry
}

这样,变量首先将被初始化为nil(函数类型为零值),这意味着它们不依赖于DownloadFile()andDoRetry()函数,稍后当包init()函数运行时,会将函数值分配给变量,因此没有初始化循环会发生。

于 2018-08-03T07:49:10.177 回答