template
template
官方文档:template package - text/template - Go Packages
在平时我们经常会使用fmt.Sprintf
函数来进行字符串格式化,但它只适用于处理小字符串的情况,而且需要使用格式化动词来指定类型,无法做到参数命名,不支持复杂情况下的处理,而这就是模板引擎所需要解决的问题,比如在直接挂到后端的静态HTML
页面就需要用到模板引擎。社区里面有很多优秀的第三方模板引擎库,比如pongo2
,sprig
,jet
,不过本文要讲述的主角是 go 内置的模板引擎库text/template
,在实际开发中一般用的是html/template
,后者基于前者并做了很多关于HTML
的安全处理,一般情况使用前者即可,若是涉及到HTML
的模板处理建议使用后者会更安全。
快速开始
下面来看一个关于模板引擎的简单使用示例,如下所示
package main
import (
"fmt"
"os"
"text/template"
)
func main() {
tmpl := `This is the first template string, {{ .message }}`
te, err := template.New("texTmpl").Parse(tmpl)
if err != nil {
fmt.Println(err)
return
}
data := map[string]any{
"message": "hello world!",
}
execErr := te.Execute(os.Stdout, data)
if execErr != nil {
fmt.Println(err)
}
}
上述代码的输出为
This is the first template string, hello world!
在案例代码中,tmpl
是一个模板字符串,字符串中的{{ .message }}
是模板引擎的模板参数。首先通过*Template.Parse
方法解析模板字符串,
func (t *Template) Parse(text string) (*Template, error)
解析成功后再通过*Template.Execute
方法将data
数据应用于模板中,最后输出到传入的Writer
中也就是os.Stdout
。
func (t *Template) Execute(wr io.Writer, data any) error
在以后模板引擎的使用中,基本上都是这三步:
- 获取模板
- 解析模板,
- 将数据应用到模板中
可见模板引擎的使用其实相当简单,稍微复杂一点的是模板引擎的模板语法,这才是本文主要讲解的内容。
模板语法
参数
go 通过两对花括号{{ }}
,来在模板中表示这是一个模板参数,通过.
来表示根对象,根对象就是传入的data
。就像是访问一个类型的成员变量一样,通过.
符号衔接变量名就可以在模板中访问对应的值,例如
{{ .data }}
前提与之同名的成员变量存在,否则就会报错。对于传入的data
,一般是结构体或者map
,也可以是基本类型,比如数字字符串,这时.
所代表的根对象就是其自身。在花括号内,不一定非得去访问根对象来获取值,也可以是基本类型的字面量,例如
{{ 1 }}
{{ 3.14 }}
{{ "jack" }}
不管什么类型,最终都会通过fmt.Sprintf("%s", val)
来获取其字符串表现形式,看下面的例子。
func main() {
out := os.Stdout
tmpl := "data-> {{ . }}\n"
datas := []any{
"hello world!",
6379,
3.1415926,
[]any{1, "2*2", 3.6},
map[string]any{"data": "hello world!"},
struct {
Data string
}{Data: "hello world!"},
}
for _, data := range datas {
err := ExecTmpl(out, tmpl, data)
if err != nil {
panic(err)
}
}
}
func ExecTmpl(writer io.Writer, tmpl string, data any) error {
parsedTmpl, err := template.New("template").Parse(tmpl)
if err != nil {
return err
}
return parsedTmpl.Execute(writer, data)
}
输出如下
data-> hello world!
data-> 6379
data-> 3.1415926
data-> [1 2*2 3.6]
data-> map[data:hello world!]
data-> {hello world!}
可以看到其输出形式跟直接使用fmt.Sprintf
一致。对于结构体和 map,可以通过字段名来访问其值,如下所示
func main() {
out := os.Stdout
tmpl := "data-> {{ .Data }}\n"
datas := []any{
map[string]any{"Data": "hello world!"},
struct {
Data string
}{Data: "hello world!"},
}
for _, data := range datas {
err := ExecTmpl(out, tmpl, data)
if err != nil {
panic(err)
}
}
}
输出如下
data-> hello world!
data-> hello world!
对于切片和map
,虽然并没有提供特定语法来访问某一个索引的值,但可以通过函数调用的方式来实现,如下所示
func main() {
out := os.Stdout
tmpl := "data-> {{ index . 1}}\n"
datas := []any{
[]any{"first", "second"},
map[int]any{1: "first"},
}
for _, data := range datas {
err := ExecTmpl(out, tmpl, data)
if err != nil {
panic(err)
}
}
}
输出
data-> second
data-> first
如果是多维切片,可以通过如下方式来访问对应下标的值,等同于s[i][j][k]
{{ index . i j k }}
对于嵌套的结构体或 map,可以使用.k1.k2.k3
这种方式访问,例如
{{ .person.father.name }}
在使用模板参数时,可以在参数前后加上-
符号来消除参数前后的空白,看个例子
func main() {
out := os.Stdout
tmpl := `{{ .x }} {{ - .op - }} {{ .y }}`
datas := []any{
map[string]any{"x": "10", "op": ">", "y": "2"},
}
for _, data := range datas {
err := ExecTmpl(out, tmpl, data)
if err != nil {
panic(err)
}
}
}
正常来说输出结果应该是10 > 2
,但由于在op
参数前后添加了-
符号,所以它前后的空白符都会被消除,所以实际输出为
10>2
需要注意的是,在花括号中,-
符号与参数必须相隔一个空格,也就说必须是{{- . -}}
这种格式,在例子中之所以会在两边额外加个空格写成{{ - . - }}
这种格式纯粹是个人觉得看的顺眼,实际上并没有这个语法限制。
注释
模板语法支持注释,注释并不会在最终的模板中生成,其语法如下
{{/* this is a comment */}}
注释符号/*
和*/
必须与花括号相邻,它们之间不能有其它字符,否则将无法正常解析。只有一种情况例外,那就是消除空白符的时候
{{- /* this is a comment */ -}}
变量
在模板中也可以声明变量,通过$
符号来表示这是一个变量,并通过:=
来进行赋值,就跟 go 代码一样,例子如下。
{{ $name := .Name }}
{{ $val := index . 1 }}
{{ $val := index .dict key }}
// 整型赋值
{{ $numer := 1 }}
// 浮点数赋值
{{ $float := 1.234}}
// 字符串赋值
{{ $name := "jack" }}
在后续使用时,通过$
衔接变量名来访问该变量的值,比如
func main() {
out := os.Stdout
tmpl := `{{ $name := .name }} {{- $name }}`
datas := []any{
map[string]any{"name": "jack"},
}
for _, data := range datas {
err := ExecTmpl(out, tmpl, data)
if err != nil {
panic(err)
}
}
}
输出
jack
变量必须先声明才能使用,否则将会提示undefined variable
,并且也要在作用域内才能使用。
函数
模板自身的语法其实并不多,大多数功能都是通过函数来实现的,函数调用的格式为函数名后衔接参数列表,以空格为分隔符,如下所示
{{ funcname arg1 arg2 arg3 ... }}
例如之前用到的index
函数
{{ index .s 1 }}
用于比较是否相等的函数eq
函数
{{ eq 1 2 }}
每一个*Template
都有一个FuncsMap
,用于记录函数的映射
type FuncMap map[string]any
在创建模板时从text/template.builtins
获取默认的函数映射表,下面是内置的所有函数
函数名 | 作用 | 示例 |
---|---|---|
and | 与运算 | {{ and true false }} |
or | 或运算 | {{ or true false }} |
not | 取反运算 | {{ not true }} |
eq | 是否相等 | {{ eq 1 2 }} |
ne | 是否不相等 | {{ ne 1 2 }} |
lt | 小于 | {{ lt 1 2 }} |
le | 小于等于 | {{ le 1 2 }} |
gt | 大于 | {{ gt 1 2 }} |
ge | 大于等于 | {{ ge 1 2 }} |
len | 返回长度 | {{ len .slice }} |
index | 获取目标指定索引的元素 | {{ index . 0 }} |
slice | 切片,等价于 s[1:2:3] | {{ slice . 1 2 3 }} |
html | HTML 转义 | {{ html .name }} |
js | js 转义 | {{ js .name }} |
print | fmt.Sprint | {{ print . }} |
printf | fmt.Sprintf | {{ printf "%s" .}} |
println | fmt.Sprintln | {{ println . }} |
urlquery | url query 转义 | {{ urlquery .query }} |
除了这些之外,还有一个比较特殊的内置函数call
,它是用于直接调用通过在Execute
时期传入的data
中的函数,例如下面的模板
{{ call .string 1024 }}
传入的数据如下
map[string]any{
"string": func(val any) string { return fmt.Sprintf("%v: 2048", val) },
}
那么在模板中就会生成
1024: 2048
这是自定义函数的途径之一,不过通常建议使用*Template.Funcs
方法来添加自定义函数,因为后者可以作用全局,不需要绑定到根对象中。
func (t *Template) Funcs(funcMap FuncMap) *Template
自定义函数的返回值一般有两个,第一个是需要用到的返回值,第二个是error
。例如有如下自定义函数
template.FuncMap{
"add": func(val any) (string, error) { return fmt.Sprintf("%v+1", val), nil },
}
然后在模板中直接使用
{{ add 1024 }}
其结果为
1024 + 1
管道
这个管道与chan
是两个东西,官方文档里面称其为pipeline
,任何能够产生数据的操作都称其为pipeline
。下面的模板操作都属于管道操作
{{ 1 }}
{{ eq 1 2 }}
{{ $name }}
{{ .name }}
熟悉 linux 的应该都知道管道运算符|
,模板中也支持这样的写法。管道操作在模板中经常出现,例如
{{ $name := 1 }}{{ $name | print | printf "%s+1=?" }}
其结果为
1+1=?
在后续的with
,if
,range
中也会频繁用到。
with
通过with
语句可以控制变量和根对象的作用域,格式如下
{{ with pipeline }}
text
{{ end }}
with
会检查管道操作返回的值,如果值为空的话,中间的text
模板就不会生成。如果想要处理空的情况,可以使用with else
,格式如下
{{ with pipeline }}
text1
{{ else }}
text2
{{ end }}
如果管道操作返回的值为空,那么就会执行else
这块的逻辑。在with
语句中声明的变量,其作用域仅限于with
语句内,看下面一个例子
{{ $name := "mike" }}
{{ with $name := "jack" }}
{{- $name -}}
{{ end }}
{{- $name -}}
它的输出如下,显然这是由于作用域不同,它们是两个不同的变量。
jackmike
通过with
语句还可以在作用域内改写根对象,如下
{{ with .name }}
name: {{- .second }}-{{ .first -}}
{{ end }}
age: {{ .age }}
address: {{ .address }}
传入如下的数据
map[string]any{
"name": map[string]any{
"first": "jack",
"second": "bob",
},
"age": 1,
"address": "usa",
}
它的输出
name:bob-jack
age: 1
address: usa
可以看到在with
语句内部,根对象.
已经变成了.name
。
条件
条件语句的格式如下所示
{{ if pipeline }}
text1
{{ else if pipeline }}
text2
{{ else }}
text3
{{ end }}
就跟写普通的代码一样,非常好理解。下面看几个简单的例子,
{{ if eq .lang "en" }}
{{- .content.en -}}
{{ else if eq .lang "zh" }}
{{- .content.zh -}}
{{ else }}
{{- .content.fallback -}}
{{ end }}
传入的数据
map[string]any{
"lang": "zh",
"content": map[string]any{
"en": "hello, world!",
"zh": "你好,世界!",
"fallback": "hello, world!",
},
}
例子中的模板根据传入的语言lang
来决定要以何种方式展示内容,输出结果
你好,世界!
迭代
迭代语句的格式如下,range
所支持的pipeline
必须是数组,切片,map
,以及channel
。
{{ range pipeline }}
loop body
{{ end }}
结合else
使用,当长度为 0 时,就会执行else
块的内容。
{{ range pipeline }}
loop body
{{ else }}
fallback
{{ end }}
除此之外,还支持break
,continue
这类操作,比如
{{ range pipeline }}
{{ if pipeline }}
{{ break }}
{{ end }}
{{ if pipeline }}
{{ continue }}
{{ end }}
loop body
{{ end }}
下面看一个迭代的例子。
{{ range $index, $val := . }}
{{- if eq $index 0 }}
{{- continue -}}
{{ end -}}
{{- $index}}: {{ $val }}
{{ end }}
传入数据
[]any{1, "2", 3.14},
输出
1: 2
2: 3.14
迭代map
也是同理。
嵌套
一个模板中可以定义有多个模板,比如
{{ define "t1" }} t1 {{ end }}
{{ define "t2" }} t2 {{ end }}
这些定义的模板在并不会生成在最终的模板中,除非在加载时指定了名称或者通过template
语句手动指定。
func (t *Template) ExecuteTemplate(wr io.Writer, name string, data any) error
比如下面的例子
{{ define "t1" }}
{{- with .t1 }}
{{- .data -}}
{{ end -}}
{{ end }}
{{ define "t2" }}
{{- with .t2 }}
{{- .data -}}
{{ end}}
{{ end -}}
传入如下数据
map[string]any{
"t1": map[string]any{"data": "template body 1"},
"t2": map[string]any{"data": "template body 2"},
}
代码
func main() {
out := os.Stdout
tmpl :=
`{{ define "t1" }}
{{- with .t1 }}
{{- .data -}}
{{ end -}}
{{ end }}
{{ define "t2" }}
{{- with .t2 }}
{{- .data -}}
{{ end}}
{{ end -}}`
datas := []any{
map[string]any{
"t1": map[string]any{"data": "template body 1"},
"t2": map[string]any{"data": "template body 2"},
},
}
name := "t1"
for _, data := range datas {
err := ExecTmpl(out, tmpl, name, data)
if err != nil {
panic(err)
}
}
}
func ExecTmpl(writer io.Writer, tmpl string, name string, data any) error {
t := template.New("template")
parsedTmpl, err := t.Parse(tmpl)
if err != nil {
return err
}
return parsedTmpl.ExecuteTemplate(writer, name, data)
}
输出
template body 1
或者也可以手动指定模板
{{ define "t1" }}
{{- with .t1 }}
{{- .data -}}
{{ end -}}
{{ end }}
{{ define "t2" }}
{{- with .t2 }}
{{- .data -}}
{{ end}}
{{ end -}}
{{ template "t2" .}}
那么在解析时是否指定模板名称,t2 都会加载。
关联
子模板只是在一个模板内部声明多个命名的模板,关联是将外部的多个命名的*Template
关联起来。然后通过template
语句来引用指定的模板。
{{ tempalte "templateName" pipeline}}
pipeline
可以根据自己的需求来指定关联模板的根对象,或者也可以直接传入当前模板的根对象。看下面的一段代码例子
func main() {
tmpl1 := `name: {{ .name }}`
tmpl2 := `age: {{ .age }}`
tmpl3 := `Person Info
{{template "t1" .}}
{{template "t2" .}}`
t1, err := template.New("t1").Parse(tmpl1)
if err != nil {
panic(err)
}
t2, err := template.New("t2").Parse(tmpl2)
if err != nil {
panic(err)
}
t3, err := template.New("t3").Parse(tmpl3)
if err != nil {
panic(err)
}
if err := associate(t3, t1, t2); err != nil {
panic(err)
}
err = t3.Execute(os.Stdout, map[string]any{
"name": "jack",
"age": 18,
})
if err != nil {
panic(err)
}
}
func associate(t *template.Template, ts ...*template.Template) error {
for _, tt := range ts {
_, err := t.AddParseTree(tt.Name(), tt.Tree)
if err != nil {
return err
}
}
return nil
}
在上述的地面中,t3 关联了 t1,和 t2,使用*Template.AddParseTree
方法进行关联
func (t *Template) AddParseTree(name string, tree *parse.Tree) (*Template, error)
最终的模板生成结果为
Person Info
name: jack
age: 18
插槽
通过block
语句,可以实现类似 vue 插槽的效果,其目的是为了复用某一个模板而用的。看一个使用案例就知道怎么用了,在 t1 模板中定义插槽
Basic Person Info
name: {{ .name }}
age: {{ .age }}
address: {{ .address }}
{{ block "slot" . }} default content body {{ end }}
block
语句可以插槽中的默认内容,在后续其它模板使用插槽时,会覆盖默认的内容。在 t2 模板中引用 t1 模板,并使用define
定义嵌入的内容
{{ template "person.txt" . }}
{{ define "slot" }}
school: {{ .school }}
{{ end }}
将两个模板关联以后,传入如下的数据
map[string]any{
"name": "jack",
"age": 18,
"address": "usa",
"company": "google",
"school": "mit",
}
最终输出的结果为
Basic Person Info
name: jack
age: 18
address: usa
school: mit
模板文件
在模板语法的案例中,都是使用的字符串字面量来作为模板,在实际的使用情况中大多数都是将模板放在文件中。
func ParseFS(fsys fs.FS, patterns ...string) (*Template, error)
比如template.ParseFs
就是从指定的文件系统中加载匹配pattern
的模板。下面的例子以embed.FS
作为文件系统,准备三个文件
# person.txt
Basic Person Info
name: {{ .name }}
age: {{ .age }}
address: {{ .address }}
{{ block "slot" . }} {{ end }}
# student.txt
{{ template "person.txt" . }}
{{ define "slot" }}
school: {{ .school }}
{{ end }}
# employee.txt
{{ template "person.txt" . }}
{{ define "slot" }}
company: {{ .company }}
{{ end }}
代码如下
import (
"embed"
"os"
"text/template"
)
//go:embed *.txt
var fs embed.FS
func main() {
data := map[string]any{
"name": "jack",
"age": 18,
"address": "usa",
"company": "google",
"school": "mit",
}
t1, err := template.ParseFS(fs, "person.txt", "student.txt")
if err != nil {
panic(err)
}
t1.Execute(os.Stdout, data)
t2, err := template.ParseFS(fs, "person.txt", "employee.txt")
if err != nil {
panic(err)
}
t2.Execute(os.Stdout, data)
}
输出为
Basic Person Info
name: jack
age: 18
address: usa
school: mit
Basic Person Info
name: jack
age: 18
address: usa
company: google
这是一个很简单的模板文件使用案例,person.txt
作为插槽文件,其它两个复用其内容并嵌入自定义的新内容。也可以使用下面两个函数
func ParseGlob(pattern string) (*Template, error)
func ParseFiles(filenames ...string) (*Template, error)
ParseGlob
基于通配符匹配,ParseFiles
基于文件名,它们都是使用的本地文件系统。如果是用于展示在前端的html
文件,建议使用html/template
包,它提供的 API 与text/template
完全一致,但是针对html
,css
,js
做了安全处理。