东岳网络工作室

Go: tasks repeated at intervals

codeworm96 Go

Suppose we would like to perform a task repeatedly at regular intervals with a goroutine.

Here is the first try:

package main

import (
	"fmt"
	"time"
)

func main() {
	go func() {
		for {
			fmt.Println("hello, world!")
			time.Sleep(5 * time.Second)
		}
	}()
}

Oops, it does not work. The main function terminates immediately after starting the goroutine, and the program terminates with the termination of the main function.

We should let the main function wait for the goroutine. Therefore, WaitGroup is needed.

package main

import (
	"fmt"
	"sync"
	"time"
)

type WaitGroupWrapper struct {
	wg sync.WaitGroup
}

func (w *WaitGroupWrapper) Spawn(f func()) {
	w.wg.Add(1)
	go func() {
		f()
		w.wg.Done()
	}()
}

func (w *WaitGroupWrapper) Wait() {
	w.wg.Wait()
}

func main() {
	var wg WaitGroupWrapper
	wg.Spawn(func() {
		for {
			fmt.Println("hello, world!")
			time.Sleep(5 * time.Second)
		}
	})
	wg.Wait()
}

Note that I used a wrapper to wrap the WaitGroup to make it easier to use. Now the program works, but we can improve it with tickers, which provide by Go for doing something repeatedly at regular intervals.

ticker := time.NewTicker(5 * time.Second)
wg.Spawn(func() {
    for {
        <-ticker.C
        fmt.Println("hello, world!")
    }
})

Currently, our goroutine runs forever. We can stop it using another channel.

func main() {
	var wg WaitGroupWrapper
	done := make(chan struct{})
	ticker := time.NewTicker(5 * time.Second)
	wg.Spawn(func() {
		for {
			select {
			case <-ticker.C:
				fmt.Println("hello, world!")
			case <-done:
				goto exit
			}
		}
	exit:
	})
	time.Sleep(16 * time.Second)
	close(done)
	wg.Wait()
}

License

codeworm96
想做 infra