Go 时间处理入门:time 包、时区、定时器和超时控制怎么用

本文讲解 Go time 包中的时间解析、格式化、时区、Duration、Ticker、Timer 和超时控制,帮助初学者处理真实业务时间问题。

时间问题总比看起来更复杂

业务代码里到处都有时间:用户注册时间、订单过期时间、活动开始时间、接口超时、定时任务、缓存有效期、日志时间戳。刚开始你可能只会用 time.Now(),但很快会遇到格式化、解析、时区和定时器。时间处理一旦写错,问题往往很隐蔽:本地正常,线上差 8 小时;今天正常,月底出错;测试偶尔超时,查不出原因。

Go 的 time 包设计得很完整。它有 time.Time 表示时间点,time.Duration 表示时间长度,time.Location 表示时区,TimerTicker 处理定时。掌握这些基本概念,就能覆盖大多数入门和中小项目需求。

这篇文章会用订单过期、日志格式化、活动时间判断和定时任务做例子,讲清楚 Go 时间处理的常见写法。

time.Time 表示一个时间点

当前时间:

now := time.Now()
fmt.Println(now)

获取 Unix 时间戳:

fmt.Println(now.Unix())
fmt.Println(now.UnixMilli())

创建指定时间:

start := time.Date(2020, 3, 11, 10, 0, 0, 0, time.Local)
fmt.Println(start)

time.Date 的参数分别是年、月、日、时、分、秒、纳秒和时区。time.Local 表示本地时区。服务端程序最好明确时区,不要让不同机器的本地配置影响业务。

计算时间差:

deadline := now.Add(30 * time.Minute)
fmt.Println(deadline.Sub(now))

Add 加上一段时间,Sub 计算两个时间点之间的差,返回 time.Duration

Duration 表示时间长度

time.Duration 本质是纳秒数量,但你很少直接写数字,而是用常量组合:

timeout := 3 * time.Second
cacheTTL := 15 * time.Minute
retention := 7 * 24 * time.Hour

不要写:

timeout := 3000

这完全看不出单位。Go 的时间常量让代码自解释。

业务例子:订单 30 分钟内未支付就过期。

type Order struct {
	ID        int64
	CreatedAt time.Time
	Paid      bool
}

func (o Order) Expired(now time.Time) bool {
	if o.Paid {
		return false
	}
	return now.Sub(o.CreatedAt) > 30*time.Minute
}

注意函数接收 now time.Time,而不是内部直接调用 time.Now()。这样测试更容易:

created := time.Date(2020, 3, 11, 10, 0, 0, 0, time.UTC)
order := Order{CreatedAt: created}

if !order.Expired(created.Add(31 * time.Minute)) {
	t.Fatal("order should be expired")
}

把当前时间作为参数传入,是时间相关代码很实用的测试技巧。

Go 的时间格式化很特别

Go 格式化时间不用 YYYY-MM-DD,而是用一个固定参考时间:

formatted := now.Format("2006-01-02 15:04:05")
fmt.Println(formatted)

这个参考时间是:

Mon Jan 2 15:04:05 MST 2006

你可以把它记成 2006-01-02 15:04:05。虽然第一次看很怪,但用久了会发现它避免了 %Y%m%d 这类占位符记忆。

解析时间:

value := "2020-03-11 10:30:00"
t, err := time.ParseInLocation("2006-01-02 15:04:05", value, time.Local)
if err != nil {
	fmt.Println("parse time:", err)
	return
}
fmt.Println(t)

time.Parse 默认按 UTC 解析没有时区的信息。处理本地业务时间时,更常用 time.ParseInLocation,明确告诉它按哪个时区理解。

常见日期格式:

const dateLayout = "2006-01-02"
const dateTimeLayout = "2006-01-02 15:04:05"

把 layout 定义成常量,可以避免到处手写。

时区要显式

加载时区:

loc, err := time.LoadLocation("Asia/Shanghai")
if err != nil {
	return err
}

解析上海时间:

t, err := time.ParseInLocation("2006-01-02 15:04:05", "2020-03-11 10:00:00", loc)

转换到 UTC:

fmt.Println(t.UTC())

很多系统会选择“存储用 UTC,展示按用户时区”。这是一条很稳的经验。数据库里存 UTC,可以减少跨地区和夏令时问题;展示时再转换成用户所在时区。

不要随便把时间格式化成字符串后再比较:

if start.Format(dateTimeLayout) < end.Format(dateTimeLayout) {
}

应该直接比较 time.Time

if start.Before(end) {
}

时间类型已经知道如何比较,不需要绕成字符串。

Timer 和 Ticker

time.Timer 表示一次性定时:

timer := time.NewTimer(2 * time.Second)
<-timer.C
fmt.Println("time is up")

更常见的简单写法:

<-time.After(2 * time.Second)

time.After 会返回一个 channel,到时间后收到值。它适合简单超时,但在循环里大量使用时要注意资源。复杂场景用 Timer 更可控。

time.Ticker 表示周期性触发:

ticker := time.NewTicker(1 * time.Second)
defer ticker.Stop()

for i := 0; i < 3; i++ {
	<-ticker.C
	fmt.Println("tick", i)
}

一定要 Stop,否则 ticker 会继续占用资源。

定时任务常见结构:

func runEvery(ctx context.Context, interval time.Duration, job func()) {
	ticker := time.NewTicker(interval)
	defer ticker.Stop()

	for {
		select {
		case <-ticker.C:
			job()
		case <-ctx.Done():
			return
		}
	}
}

这段代码有退出路径。服务停止时取消 context,定时任务就能结束。不要写一个永远没有取消机制的 goroutine。

用时间控制外部调用超时

HTTP 客户端可以设置超时:

client := &http.Client{
	Timeout: 3 * time.Second,
}

resp, err := client.Get("https://example.com")
if err != nil {
	return err
}
defer resp.Body.Close()

更灵活的是使用 context:

ctx, cancel := context.WithTimeout(context.Background(), 2*time.Second)
defer cancel()

req, err := http.NewRequestWithContext(ctx, http.MethodGet, "https://example.com", nil)
if err != nil {
	return err
}

resp, err := http.DefaultClient.Do(req)
if err != nil {
	return err
}
defer resp.Body.Close()

服务端代码里通常使用请求自带的 context:

ctx, cancel := context.WithTimeout(r.Context(), 2*time.Second)
defer cancel()

这样客户端断开、服务端超时和内部调用超时都能沿着 context 传播。

小结

Go 时间处理的几个核心概念要分清:time.Time 是时间点,time.Duration 是时间长度,time.Location 是时区,Timer 是一次性定时,Ticker 是周期触发。格式化和解析使用固定参考时间 2006-01-02 15:04:05,不要套用其他语言的格式占位符。

真实业务里,时间代码要特别重视可测试性和时区。需要判断当前时间时,优先把 now 作为参数传入;存储时间尽量用 UTC,展示时再转时区;定时器和 ticker 要有停止机制;外部调用要设置超时。

时间问题很少在 happy path 暴露,通常在部署环境、跨时区、月底和高并发时出现。入门阶段把这些基本习惯养好,后面会少踩很多坑。

继续阅读

探索更多技术文章

浏览归档,发现更多关于系统设计、工具链和工程实践的内容。

全部文章 返回首页