为了测试并发的gorout例程,我在函数中添加了一行,使其返回的时间随机(最多一秒)

time.Sleep(rand.Int31n(1000) * time.Millisecond)

然而,当我编译时,我得到了这个错误

\履带。go:49:无效操作:rand.Int31n(1000) *时间。毫秒(类型int32和time.Duration不匹配)

什么好主意吗?如何乘以持续时间?


当前回答

Go有一个Duration类型,这很好——显式定义的单位可以防止现实世界的问题。

由于Go的严格类型规则,您不能将Duration与整数相乘——必须使用强制转换才能将公共类型相乘。

/*
MultiplyDuration Hide semantically invalid duration math behind a function
*/
func MultiplyDuration(factor int64, d time.Duration) time.Duration {
    return time.Duration(factor) * d        // method 1 -- multiply in 'Duration'
 // return time.Duration(factor * int64(d)) // method 2 -- multiply in 'int64'
}

官方文档演示了使用方法#1:

要将一个整数单位转换为持续时间,请乘以:

seconds := 10
fmt.Print(time.Duration(seconds)*time.Second) // prints 10s

但是,当然,一个持续时间乘以另一个持续时间不应该得到一个持续时间,从表面上看,这是荒谬的。举个例子,5毫秒乘以5毫秒产生6h56m40。试图平方5秒会导致溢出(如果使用常量,甚至不会编译)。

顺便说一下,以纳秒为单位的Duration的int64表示形式“将最大可表示的持续时间限制在大约290年”,这表明Duration与int64一样,被视为有符号值:(1<<(64-1))/(1e9*60*60*24*365.25) ~= 292,这正是它的实现方式:

// A Duration represents the elapsed time between two instants
// as an int64 nanosecond count. The representation limits the
// largest representable duration to approximately 290 years.
type Duration int64

因此,因为我们知道Duration的底层表示是一个int64,所以在int64和Duration之间执行强制转换是一个合理的no - op——只需要满足关于混合类型的语言规则,它对后续的乘法操作没有影响。

如果出于纯粹的原因不喜欢这种类型转换,就像我上面所示的那样,将它隐藏在一个函数调用中。

其他回答

Int32和时间。持续时间是不同的类型。您需要将int32转换为时间。持续时间:

time.Sleep(time.Duration(rand.Int31n(1000)) * time.Millisecond)

您必须将其转换为正确的格式。

yourTime := rand.Int31n(1000)
time.Sleep(time.Duration(yourTime) * time.Millisecond)

如果您将检查sleep的文档,您将看到它需要func sleep (d Duration) Duration作为参数。你的兰德。Int31n返回int32。

示例中的行工作(时间。Sleep(100 * time.Millisecond)),因为编译器足够聪明,可以理解这里常量100表示持续时间。但如果你传递一个变量,你应该强制转换它。

你可以使用time.ParseDuration。

ms := rand.Int31n(1000)
duration, err := time.ParseDuration(fmt.Sprintf(
    "%vms",
    ms,
))

被一些关于Duration与Duration相乘的评论和讨论所迷惑,我玩了一些单位和函数,得到了这个:

time.Second =  1s
time.Minute =  1m0s

time.Duration(1) = 1ns
time.Duration(1) * time.Millisecond =  1ms
time.Duration(1) * time.Second =  1s
time.Duration(1) * time.Minute =  1m0s

在Go中,你可以将相同类型的变量相乘,所以你需要让表达式的两个部分具有相同的类型。

您可以做的最简单的事情是在相乘之前将一个整数转换为duration,但这将违反单元语义。持续时间乘以持续时间的单位是多少?

我宁愿转换时间。毫秒转换为int64,然后将其乘以毫秒数,然后转换为time。持续时间:

time.Duration(int64(time.Millisecond) * int64(rand.Int31n(1000)))

这样,表达式的任何部分根据其类型都可以说具有有意义的值。int64(time. millisecond)部分只是一个无量纲值——原始值中最小时间单位的数量。

如果走一条稍微简单的路:

time.Duration(rand.Int31n(1000)) * time.Millisecond

乘法运算的左边部分是无意义的——一个“time”类型的值。Duration”,持有与其类型无关的东西:

numberOfMilliseconds := 100
// just can't come up with a name for following:
someLHS := time.Duration(numberOfMilliseconds)
fmt.Println(someLHS)
fmt.Println(someLHS*time.Millisecond)

这不仅仅是语义,还有与类型相关的实际功能。 这段代码输出:

100ns
100ms

有趣的是,这里的代码示例使用了最简单的代码,具有相同的Duration转换语义:https://golang.org/pkg/time/#Duration

秒数:= 10 fmt.Print(time.Duration(seconds)*time.Second) //打印10秒