在 go 模板中,`{{ $var := value }}` 在 `range` 内部会**重新声明同名变量**而非赋值,导致其作用域仅限当次迭代,无法跨循环更新状态。go 1.11+ 支持 `{{ $var = value }}` 赋值语法,但旧版本需通过索引函数或方法绕过限制。
Go 模板的变量作用域规则是理解该问题的关键:模板中所有 := 都是变量声明(declaration),而非赋值(assignment)。这意味着:
注册一个辅助函数,在模板中根据当前索引获取前一项日期:
// Go 代码:注册函数
func prevDate(posts []Post, i int) string {
if i <= 0 || i >= len(posts) {
return ""
}
return posts[i-1].Date
}
tmpl := template.Must(template.New("blog").
Funcs(map[string]interface{}{
"PrevDate": prevDate,
}).
Parse(`{{ range $i, $post := .Posts }}
{{ $prevDate := PrevDate $.Posts $i }}
{{ if ne $prevDate $post.Date }}
Posts dated: {{ $post.Date }}
{{ end }}
{{ $post.Content }}
{{ end }}`))⚠️ 注意:函数参数顺序需与模板调用一致(PrevDate $.Posts $i),$.Posts 确保传入原始切片。
更面向对象的方式——为 Posts 类型添加 PrevDate(i int) 方法:
type Post struct {
Content string
Date string
}
type Posts []Post
func (p Posts) PrevDate(i int) string {
if i <= 0 || i >= len(p) {
return ""
}
return p[i-1].Date
}模板中直接调用:
{{ range $i, $post := .Pos
ts }}
{{ $prevDate := $.Posts.PrevDate $i }}
{{ if ne $prevDate $post.Date }}
Posts dated: {{ $post.Date }}
{{ end }}
{{ $post.Content }}
{{ end }}若项目可升级至 Go 1.11 或更高版本,可直接使用赋值语法 =(非 :=):
{{ $prevDate := "" }}
{{ range $post := .Posts }}
{{ if ne $prevDate $post.Date }}
Posts dated: {{ $post.Date }}
{{ end }}
{{ $post.Content }}
{{ $prevDate = $post.Date }} {{/* 注意:此处是 =,不是 := */}}
{{ end }}✅ 此时 $prevDate = ... 是真正的变量更新,作用域保持不变,可跨迭代生效。