golang执行功能恰好在1秒钟内每1毫秒(每秒1000次调用)

时间:2018-06-22 19:31:59

标签: go milliseconds

我是golang的新手。

我想在某个持续时间(现在为1秒)内恰好在1毫秒内调用一个函数,该函数应每秒调用1000次。

我确实遵循并实现了使用timer和tick的多种方法,但是当在一个简单的函数上进行测试时,每次调用它仅将一个值增加一个,我发现它仅在一秒钟内被调用了约630次。

当我循环调用该函数并在1秒钟后将其停止时,该函数在1秒钟内被称为〜226370427时间,这意味着我在计时器和刻度上有问题。

这是我使用goroutine并发实现的代码,该代码不断调用使用代码运行器每1毫秒递增一次的函数,并在1秒后由main停止,然后使用int通道读取总和值。

import (
    "fmt"
    "time"
)

func main() {
    tickChan := time.NewTicker(time.Millisecond * 1).C

    doneChan := make(chan bool)
    resultChan := make(chan int)
    go start(tickChan, doneChan, resultChan)
    time.Sleep(time.Second * 1)
    doneChan <- true // Set it to true to stop the infinte loop in the routine
    fmt.Printf("total is : %v\n", <-resultChan)

}

func start(tickChan <-chan time.Time, doneChan <-chan bool, resultChan chan int) {
    sum := 0 // <-- The the value to increment
    for {
        select {
        case <-tickChan:
            sum++
        case <-doneChan:
            resultChan <- sum
            return
        }
    }
}

我能够在node.js中成功地做到这一点(使用nanotimer lib),并且能够每1毫秒在1秒内精确地运行1000次,我什至也可以在微秒内完成它(大约是988676)在1秒内(而不是100万))

编辑:

这是我的Go版本go version go1.10.2 windows/amd64

然后去env `

set GOARCH=amd64
set GOBIN=
set GOCACHE=C:\Users\name\AppData\Local\go-build
set GOEXE=.exe
set GOHOSTARCH=amd64
set GOHOSTOS=windows
set GOOS=windows
set GOPATH=C:\Users\name\Documents\Go Projects
set GORACE=
set GOROOT=C:\Go
set GOTMPDIR=
set GOTOOLDIR=C:\Go\pkg\tool\windows_amd64
set GCCGO=gccgo
set CC=gcc
set CXX=g++
set CGO_ENABLED=1
set CGO_CFLAGS=-g -O2
set CGO_CPPFLAGS=
set CGO_CXXFLAGS=-g -O2
set CGO_FFLAGS=-g -O2
set CGO_LDFLAGS=-g -O2
set PKG_CONFIG=pkg-config
set GOGCCFLAGS=-m64 -mthreads -fno-caret-diagnostics -Qunused-arguments -fmessage-length=0 -fdebug-prefix-map=C:\Users\name\AppData\Local\Temp\go-build403228958=/tmp/go-build -gno-record-gcc-switches

`

编辑2:我在操场上对其进行了测试,总共得到1000张,我认为问题出在Windows上。有什么办法可以解决?

1 个答案:

答案 0 :(得分:0)

  

Package time

import "time"
     

type Ticker

     

股票代号持有一个通道,可以在   间隔。

type Ticker struct {
        C <-chan Time // The channel on which the ticks are delivered.
        // contains filtered or unexported fields
}
     

func NewTicker

func NewTicker(d Duration) *Ticker
     

NewTicker返回一个新的Ticker,其中包含将发送   持续时间参数指定的时间。它调整   间隔或滴答滴答,以弥补接收速度慢的问题。持续时间d   必须大于零;如果没有,NewTicker将惊慌。停止   代码以释放关联的资源。


这是我在同一台双启动计算机上运行Windows 10.0.17134和Linux 4.15的结果。

package main

import (
    "fmt"
    "runtime"
    "time"
)

func ticker(tick time.Duration) {
    ticker := time.NewTicker(tick)
    defer ticker.Stop()
    done := make(chan bool)
    sleep := 1 * time.Second
    go func() {
        time.Sleep(sleep)
        done <- true
    }()
    ticks := 0
    for {
        select {
        case <-done:
            fmt.Printf("%v × %v ticks in %v\n", ticks, tick, sleep)
            return
        case <-ticker.C:
            ticks++
        }
    }
}

func main() {
    fmt.Println("GOMAXPROCS:", runtime.GOMAXPROCS(0))
    start := time.Duration(1 * time.Millisecond)
    end := start / 1000
    for tick := start; tick >= end; tick /= 10 {
        ticker(tick)
    }
}

输出:

Microsoft Windows [Version 10.0.17134.112]:

GOMAXPROCS: 4
554 × 1ms ticks in 1s
552 × 100µs ticks in 1s
542 × 10µs ticks in 1s
552 × 1µs ticks in 1s

Linux 4.15.0-23-generic:

GOMAXPROCS: 4
1000 × 1ms ticks in 1s
9990 × 100µs ticks in 1s
13169 × 10µs ticks in 1s
14014 × 1µs ticks in 1s

  

Sleep function | Microsoft Docs

     

暂停当前线程的执行,直到超时   时间间隔过去。

     

参数

dwMilliseconds
     

要暂停执行的时间间隔,以   毫秒。


在Windows上,“ Go Ticker”分辨率最多只能限制为一毫秒。