如何在Go中计时一个函数并以毫秒为单位返回其运行时间?

huangapple go评论119阅读模式
英文:

How do you time a function in Go and return its runtime in milliseconds?

问题

如何在Go中计时一个函数并以毫秒为单位返回其运行时间?

英文:

How do you time a function in Go and return its runtime in milliseconds?

答案1

得分: 38

Go的defer使得这个变得简单。

在Go 1.x中,定义以下函数:

  1. func trace(s string) (string, time.Time) {
  2. log.Println("START:", s)
  3. return s, time.Now()
  4. }
  5. func un(s string, startTime time.Time) {
  6. endTime := time.Now()
  7. log.Println(" END:", s, "ElapsedTime in seconds:", endTime.Sub(startTime))
  8. }

之后,你会得到简洁的一行经过时间的日志信息:

  1. func someFunction() {
  2. defer un(trace("SOME_ARBITRARY_STRING_SO_YOU_CAN_KEEP_TRACK"))
  3. //在这里做一些事情...
  4. }

聪明的魔法是trace()在函数开始时被调用,但un()被延迟到最后。由于日志语句的存在,它并不是原子钟精确,但如果你需要更高的精确度,这种模式是Go的一个优势。

编辑:

这个答案最初使用了旧的time包API。这里只是为了历史价值而重现:

用于Go版本在2011年12月1日之前的周刊:

  1. func trace(s string) (string, int64) {
  2. log.Println("START:", s)
  3. return s, time.Nanoseconds()
  4. }
  5. func un(s string, startTime int64) {
  6. endTime := time.Nanoseconds()
  7. log.Println(" END:", s, "ElapsedTime in seconds:", float32(endTime-startTime)/1E9)
  8. }
英文:

Go's defer makes this trivial.

In Go 1.x, define the following functions:

  1. func trace(s string) (string, time.Time) {
  2. log.Println("START:", s)
  3. return s, time.Now()
  4. }
  5. func un(s string, startTime time.Time) {
  6. endTime := time.Now()
  7. log.Println(" END:", s, "ElapsedTime in seconds:", endTime.Sub(startTime))
  8. }

After that, you get Squeaky Clean one line elapsed time log messages:

  1. func someFunction() {
  2. defer un(trace("SOME_ARBITRARY_STRING_SO_YOU_CAN_KEEP_TRACK"))
  3. //do a bunch of stuff here...
  4. }

The clever magic is that the trace() is called at the beginning of the function, but the un() is deferred to the end. It's not atomic-clock accurate, due to the log statements, but if you need more accuracy, this kind of pattern is one of Go's marshmallowy good strengths.

EDIT:

This answer originally used legacy time package API. Reproduced here for historical value only:

For use w/ Go versions prior to 12-01-2011 weekly:

  1. func trace(s string) (string, int64) {
  2. log.Println("START:", s)
  3. return s, time.Nanoseconds()
  4. }
  5. func un(s string, startTime int64) {
  6. endTime := time.Nanoseconds()
  7. log.Println(" END:", s, "ElapsedTime in seconds:", float32(endTime-startTime)/1E9)
  8. }

答案2

得分: 31

使用Go testing包来对函数进行基准测试。例如,

  1. package main
  2. import (
  3. "fmt"
  4. "testing"
  5. )
  6. // 要进行基准测试的函数
  7. func Function(n int) int64 {
  8. n64 := int64(n)
  9. return n64 * n64
  10. }
  11. func BenchmarkFunction(b *testing.B) {
  12. n := 42
  13. for i := 0; i < b.N; i++ {
  14. _ = Function(n)
  15. }
  16. }
  17. func main() {
  18. br := testing.Benchmark(BenchmarkFunction)
  19. fmt.Println(br)
  20. }

输出:

  1. 500000000 4.22 ns/op

您还可以使用Go gotest命令来运行基准测试。

英文:

Use the Go testing package to benchmark the function. For example,

  1. package main
  2. import (
  3. &quot;fmt&quot;
  4. &quot;testing&quot;
  5. )
  6. // the function to be benchmarked
  7. func Function(n int) int64 {
  8. n64 := int64(n)
  9. return n64 * n64
  10. }
  11. func BenchmarkFunction(b *testing.B) {
  12. n := 42
  13. for i := 0; i &lt; b.N; i++ {
  14. _ = Function(n)
  15. }
  16. }
  17. func main() {
  18. br := testing.Benchmark(BenchmarkFunction)
  19. fmt.Println(br)
  20. }

Output:

  1. 500000000 4.22 ns/op

You can also use the Go gotest command to run benchmarks.

答案3

得分: 17

另一种简单的方法可以是:

  1. import (
  2. "fmt"
  3. "time"
  4. )
  5. start := time.Now()
  6. // 进行一些计算
  7. elapsed := time.Since(start)
  8. fmt.Println(elapsed)

这将输出类似于 359.684612ms 的内容。

英文:

Another easy way can be:

  1. import (
  2. &quot;fmt&quot;
  3. &quot;time&quot;
  4. )
  5. start := time.Now()
  6. // some computation
  7. elapsed := time.Since(start)
  8. fmt.Println(elapsed)

which will output something like 359.684612ms

答案4

得分: 12

也许你也可以使用一个持续时间(经过的时间)来实现这个...看起来更好一些。

  1. func trace(s string) (string, time.Time) {
  2. log.Printf("跟踪开始:%s\n", s)
  3. return s, time.Now()
  4. }
  5. func un(s string, startTime time.Time) {
  6. elapsed := time.Since(startTime)
  7. log.Printf("跟踪结束:%s,经过时间 %f 秒\n", s, elapsed.Seconds())
  8. }
英文:

Perhaps you can also use a Duration (elapsed) for this...looks a little bit nicer.

  1. func trace(s string) (string, time.Time) {
  2. log.Printf(&quot;trace start: %s\n&quot;, s)
  3. return s, time.Now()
  4. }
  5. func un(s string, startTime time.Time) {
  6. elapsed := time.Since(startTime)
  7. log.Printf(&quot;trace end: %s, elapsed %f secs\n&quot;, s, elapsed.Seconds())
  8. }

答案5

得分: 1

有关时间戳和定时器的时间包有几个选项。请参阅此处的文档:http://golang.org/pkg/time/

英文:

There are several options for timestamping and timers in the time package. See the documentation here: http://golang.org/pkg/time/

huangapple
  • 本文由 发表于 2011年12月2日 09:13:37
  • 转载请务必保留本文链接:https://go.coder-hub.com/8350609.html
匿名

发表评论

匿名网友

:?: :razz: :sad: :evil: :!: :smile: :oops: :grin: :eek: :shock: :???: :cool: :lol: :mad: :twisted: :roll: :wink: :idea: :arrow: :neutral: :cry: :mrgreen:

确定