为了测试并发的gorout例程,我在函数中添加了一行,使其返回的时间随机(最多一秒)
time.Sleep(rand.Int31n(1000) * time.Millisecond)
然而,当我编译时,我得到了这个错误
\履带。go:49:无效操作:rand.Int31n(1000) *时间。毫秒(类型int32和time.Duration不匹配)
什么好主意吗?如何乘以持续时间?
为了测试并发的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——只需要满足关于混合类型的语言规则,它对后续的乘法操作没有影响。
如果出于纯粹的原因不喜欢这种类型转换,就像我上面所示的那样,将它隐藏在一个函数调用中。
其他回答
您必须将其转换为正确的格式。
yourTime := rand.Int31n(1000)
time.Sleep(time.Duration(yourTime) * time.Millisecond)
如果您将检查sleep的文档,您将看到它需要func sleep (d Duration) Duration作为参数。你的兰德。Int31n返回int32。
示例中的行工作(时间。Sleep(100 * time.Millisecond)),因为编译器足够聪明,可以理解这里常量100表示持续时间。但如果你传递一个变量,你应该强制转换它。
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——只需要满足关于混合类型的语言规则,它对后续的乘法操作没有影响。
如果出于纯粹的原因不喜欢这种类型转换,就像我上面所示的那样,将它隐藏在一个函数调用中。
在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秒
被一些关于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
你可以使用time.ParseDuration。
ms := rand.Int31n(1000)
duration, err := time.ParseDuration(fmt.Sprintf(
"%vms",
ms,
))