如何实现带有在第一个循环中立即运行的定时器的for循环机制?

How to implement a for loop mechanism with a timer that runs immediately in the first loop?

我想编写一个 golang 程序,运行作为 kubernetes 容器中的服务。该程序应该始终 运行 并且不会自行终止 - 除非出现错误。如果一个 SIGTERM 来自 kubelet / kubernetes 因为 pod 应该被删除,程序应该退出。

在我的第一种方法中,我实施了一个 for 循环,该循环应该 运行 每分钟执行一次。在这个循环中,程序从另一个服务中查询资源并对其进行处理。然后服务应该“休眠”一分钟,然后再次执行这些步骤。在“休眠”阶段,如果有 SIGTERM,程序应立即响应。

func main() {
  
  c := make(chan os.Signal, 1)
  signal.Notify(c, os.Interrupt, syscall.SIGTERM)

  //restart loop every minute 
  ticker := time.NewTicker(60 * time.Second)
  defer ticker.Stop()

  // while true loop
  for {
    select {
    case <-c:
      fmt.Println("Break the loop")
      return
    case <-ticker.C:
      fmt.Println("Hello in a loop")
    }
  }
}

我目前的方法有两个问题。首先,我第一次 运行 for 循环时必须等待一分钟。我能以某种方式配置它,使计时器仅在第一个 运行 之后 运行 秒吗?

第二个问题是程序不应在 运行 秒之间执行任何操作 - 除了对 SIGTERM 做出反应。

我不确定我解决问题的方法是否正确。我刚刚开始使用 GO 语言。也许有更好的方法来解决我的问题。

您可以在 select 语句之外完成您的工作,然后在工单通道案例中继续。这将立即执行您的工作一次,然后每次自动收报机滴答作响。

您的函数看起来不错,但最好尽可能使用 context

很多包(数据库、IO 等)都有指定上下文的选项,这通常用于定义超时。在您的情况下,将相同(或子)上下文传递给这些包将意味着它们也尊重 sigterm。

package main

import (
    "context"
    "fmt"
    "os"
    "os/signal"
    "syscall"
    "time"
)

func main() {
    ctx := cancelCtxOnSigterm(context.Background())
    startWork(ctx)
}

// cancelCtxOnSigterm returns a Context that will be cancelled when the program receives a sigterm.
func cancelCtxOnSigterm(ctx context.Context) context.Context {
    exitCh := make(chan os.Signal, 1)
    signal.Notify(exitCh, os.Interrupt, syscall.SIGTERM)

    ctx, cancel := context.WithCancel(ctx)
    go func() {
        <-exitCh
        cancel()
    }()
    return ctx
}

// startWork performs a task every 60 seconds until the context is done.
func startWork(ctx context.Context) {
    ticker := time.NewTicker(60 * time.Second)
    defer ticker.Stop()
    for {
        // Do work here so we don't need duplicate calls. It will run immediately, and again every minute as the loop continues.
        if err := work(ctx); err != nil {
            fmt.Printf("failed to do work: %s", err)
        }
        select {
        case <-ticker.C:
            continue
        case <-ctx.Done():
            return
        }
    }
}

func work(ctx context.Context) error {
    fmt.Println("doing work")
    return nil
}