text/template 渲染时变量未替换,因需显式传入导出结构体、map[string]interface{}或单值;HTML内容需用template.HTML类型避免转义。
Go 的 text/template 不会自动展开局部变量或未命名字段,必须显式传入结构体、map 或单值。常见错误是直接调用 t.Execute(os.Stdout, nil) 却期望模板里写 {{ 能生效——
nil 没有字段,自然不渲染。
type User { Name string }
{{.Name}} 一致,且 map 声明为 map[string]interface{}
{{.}},例如 t.Execute(w, "hello") 配合模板 "Say: {{.}}"
text/template 默认对所有 {{.}} 输出做 HTML 实体转义(如 → ),这是防 XSS 的默认行为。但如果你确定内容可信(比如后台拼接的 HTML 片段),不能靠 {{. | safeHTML}} ——那个函数属于 html/template,不是 text/template。
html/template 包(推荐):它识别 template.HTML 类型并跳过转义text/template:无法关闭转义,也不应尝试绕过;它本就不是为 HTML 设计的import "html/template"
t := template.Must(template.New("").Parse("{{.Body}}"))
data := struct{ Body template.HTML }{Body: template.HTML("OK")}
t.Execute(w, data)自定义函数必须在 Parse 之前注册,且函数签名必须符合 func(...interface{}) (interface{}, error) 或更具体的形式(如 func(string) string)。常见错误是注册后又调用 template.New() 创建新模板,导致函数丢失。
Funcs(map[string]interface{}),返回的是原模板对象,可链式调用{{toUpper .Name}},不是 {{toUpper(.Name)}}
func(int) string,模板里传字符串就会 panict := template.New("").Funcs(template.FuncMap{
"toUpper": strings.ToUpper,
"add": func(a, b int) int { return a + b },
})
t.Parse("{{toUpper .Text}} + {{add 2 3}}")define 定义的模板不会自动执行,必须用 {{template "name" .}} 显式调用,且传参决定子模板能访问的字段。容易忽略的是:子模板内部的 . 是你传进去的那个值,不是外层上下文。
Parse 会覆盖,建议一次性 ParseFiles 或用 Option("missingkey=error") 提前暴露问题template.Must(t.ParseGlob("*.tmpl")) 并确保 template 调用时名称完全匹配{{printf "DEBUG: %#v" .}} 查看当前作用域数据结构text/template 处理 HTML 场景。这些点一旦写错,错误信息往往不直接指向根源,得逐层检查传参路径和类型声明。