美文网首页
《Go语言四十二章经》第三十五章 模板

《Go语言四十二章经》第三十五章 模板

作者: ffhelicopter | 来源:发表于2018-09-18 20:11 被阅读153次

    《Go语言四十二章经》第三十五章 模板

    作者:李骁

    Printf也可以做到输出格式化,当然,对于简单的例子来说足够了,但是我们有时候还是需要复杂的输出格式,甚至需要将格式化代码分离开来。这时,可以使用text/template和html/template。

    Go 官方库提供了两个模板库: text/template 和 html/template 。这两个库类似,当需要输出html格式的代码时需要使用 html/template。

    35.1 text/template

    所谓模板引擎,则将模板和数据进行渲染的输出格式化后的字符程序。对于Go,执行这个流程大概需要三步。

    • 创建模板对象
    • 加载模板
    • 执行渲染模板

    其中最后一步就是把加载的字符和数据进行格式化。

    package main
    
    import (
        "log"
        "os"
        "text/template"
    )
    
    const templ = ` 
    {{range .}}----------------------------------------
    Name:   {{.Name}}
    Price:  {{.Price | printf "%4s"}}
    {{end}}`
    
    var report = template.Must(template.New("report").Parse(templ))
    
    type Book struct {
        Name  string
        Price float64
    }
    
    func main() {
        Data := []Book{{"《三国演义》", 19.82}, {"《儒林外史》", 99.09}, {"《史记》", 26.89}}
        if err := report.Execute(os.Stdout, Data); err != nil {
            log.Fatal(err)
        }
    }
    
    程序输出:
    ----------------------------------------
    Name:   《三国演义》
    Price:  %!s(float64=19.82)
    ----------------------------------------
    Name:   《儒林外史》
    Price:  %!s(float64=99.09)
    ----------------------------------------
    Name:   《史记》
    Price:  %!s(float64=26.89)
    

    如果把模板的内容存在一个文本文件里tmp.txt:

    {{range .}}----------------------------------------
    Name:   {{.Name}}
    Price:  {{.Price | printf "%4s"}}
    {{end}}
    

    我们可以这样处理:

    package main
    
    import (
        "log"
        "os"
        "text/template"
    )
    
    var report = template.Must(template.ParseFiles("tmp.txt"))
    
    type Book struct {
        Name  string
        Price float64
    }
    
    func main() {
        Data := []Book{{"《三国演义》", 19.82}, {"《儒林外史》", 99.09}, {"《史记》", 26.89}}
        if err := report.Execute(os.Stdout, Data); err != nil {
            log.Fatal(err)
        }
    }
    
    程序输出:
    
    ----------------------------------------
    Name:   《三国演义》
    Price:  %!s(float64=19.82)
    ----------------------------------------
    Name:   《儒林外史》
    Price:  %!s(float64=99.09)
    ----------------------------------------
    Name:   《史记》
    Price:  %!s(float64=26.89)
    
    Tmpl, err := template.ParseFiles("tmp.txt")  
    //建立模板,自动 new("name")
    

    ParseFiles接受一个字符串,字符串的内容是一个模板文件的路径。

    ParseGlob是用正则的方式匹配多个文件。

    假设一个目录里有a.txt b.txt c.txt的话,用ParseFiles需要写3行对应3个文件,如果有更多文件,可以用ParseGlob。

    写成template.ParseGlob("*.txt") 即可。

    var report = template.Must(template.ParseFiles("tmp.txt"))
    

    函数Must,它的作用是检测模板是否正确,例如大括号是否匹配,注释是否正确的关闭,变量是否正确的书写。

    35.2 html/template

    和text、template类似,html/template主要在提供支持HTML的功能,所以基本使用上和上面差不多,我们来看下面代码:

    index.html:

    <!doctype html>
     <head>
      <meta charset="UTF-8">
      <meta name="Author" content="">
      <meta name="Keywords" content="">
      <meta name="Description" content="">
      <title>Go</title>
     </head>
     <body>
       {{ . }} 
     </body>
    </html>
    

    main.go:

    package main
    
    import (
        "fmt"
        "net/http"
        "text/template"
    )
    
    func tHandler(w http.ResponseWriter, r *http.Request) {
        t, _ := template.ParseFiles("index.html")
        t.Execute(w, "Hello World!")
    }
    
    func main() {
        http.HandleFunc("/", tHandler)
        http.ListenAndServe(":8080", nil)
    }
    

    运行程序,在浏览器打开:http://localhost:8080/ 会看到页面显示Hello World!

    func(t *Template) ParseFiles(filenames ...string) (*Template, error)
    func(t *Template) ParseGlob(patternstring) (*Template, error)
    

    从上面代码中我们可以看到,通过ParseFile加载了单个Html模板文件。但最终的页面很可能是多个模板文件的嵌套结果。

    ParseFiles也支持加载多个模板文件,模板对象的名字则是第一个模板文件的文件名。

    ExecuteTemplate方法,用于执行指定名字的模板,下面我们根据一段代码来看看:

    Layout.html,注意在开头根据模板语法,定义了模板名字,define "layout"。
    在结尾处,通过 {{ template "index" }}

    注意:通过将模板应用于一个数据结构(即该数据结构作为模板的参数)来执行,来获得输出。模板执行时会遍历结构并将指针表示为.(称之为dot),指向运行过程中数据结构的当前位置的值。

    {{template "header" .}} 嵌套模板中,加入.dot 代表在该模板中也可以使用该数据结构,否则不能显示。

    {{ define "layout" }}
    
    <!doctype html>
     <head>
      <meta charset="UTF-8">
      <meta name="Author" content="">
      <meta name="Keywords" content="">
      <meta name="Description" content="">
      <title>Go</title>
     </head>
     <body>
       {{ . }} 
    
       {{ template "index" }}
     </body>
    </html>
    
    {{ end }}
    
    Index.html:
    
    {{ define "index" }}
    
    <div>
    <b>Index</b>
    </div>
    {{ end }}
    

    通过define定义模板,还可以通过template action引入模板,类似include。

    package main
    
    import (
        "net/http"
        "text/template"
    )
    
    func tHandler(w http.ResponseWriter, r *http.Request) {
        t, _ := template.ParseFiles("layout.html", "index.html")
        t.ExecuteTemplate(w, "layout", "Hello World!")
    }
    
    func main() {
        http.HandleFunc("/", tHandler)
        http.ListenAndServe(":8080", nil)
    }
    

    运行程序,在浏览器打开:http://localhost:8080/

    Hello World!
    Index

    有关ParseGlob方法,则是通过glob通配符加载模板,例如 t, _ := template.ParseGlob("*.html")
    

    35.3 模板语法

    【模板标签】
    模板标签用"{{"和"}}"括起来

    【变量】
    {{.}}
    此标签输出当前对象的值
    {{.Admpub}}
    表示输出Struct对象中字段或方法名称为"Admpub"的值。

    当"Admpub"是匿名字段时,可以访问其内部字段或方法, 比如"Com":{{.Admpub.Com}} ,
    如果"Com"是一个方法并返回一个Struct对象,同样也可以访问其字段或方法:{{.Admpub.Com.Field1}}

    {{.Method1 "参数值1" "参数值2"}}
    调用方法"Method1",将后面的参数值依次传递给此方法,并输出其返回值。

    {{admpub}} 此标签用于输出在模板中定义的名称为"admpub"的变量。当admpub本身是一个Struct对象时,可访问其字段:{{admpub.Field1}} 在模板中定义变量:变量名称用字母和数字组成,并带上""前缀,采用简式赋值。
    比如:{{x := "OK"}} 或 {{x := pipeline}}

    【条件判断】
    用法1:
    {{if pipeline}} T1 {{end}}
    标签结构:{{if ...}} ... {{end}}
    用法2:
    {{if pipeline}} T1 {{else}} T0 {{end}}
    标签结构:{{if ...}} ... {{else}} ... {{end}}
    用法3:
    {{if pipeline}} T1 {{else if pipeline}} T0 {{end}}
    标签结构:{{if ...}} ... {{else if ...}} ... {{end}}
    其中if后面可以是一个条件表达式(包括通道函数表达式。pipeline即通道),也可以是一个字符窜变量或布尔值变量。当为字符窜变量时,如为空字符串则判断为false,否则判断为true。

    【遍历】
    用法1:
    {{range k,v := .Var}} {{k}} => {{v}} {{end}}
    range...end结构内部如要使用外部的变量,比如.Var2,需要这样写:.Var2 (即:在外部变量名称前加符号""即可,单独的"$"意义等同于global)
    用法2:
    {{range .Var}} {{.}} {{end}}
    用法3:
    {{range pipeline}} T1 {{else}} T0 {{end}}
    当没有可遍历的值时,将执行else部分。

    【嵌入子模板】
    用法1:
    {{template "name"}}
    嵌入名称为"name"的子模板。使用前请确保已经用{{define "name"}}子模板内容{{end}}定义好了子模板内容。
    用法2:
    {{template "name" pipeline}}
    将通道的值赋给子模板中的"."(即"{{.}}")

    【子模板嵌套】
    {{define "T1"}}ONE{{end}}
    {{define "T2"}}TWO{{end}}
    {{define "T3"}}{{template "T1"}} {{template "T2"}}{{end}}
    {{template "T3"}}
    输出:
    ONE TWO

    【定义局部变量】
    用法1:
    {{with pipeline}} T1 {{end}}
    通道的值将赋给该标签内部的"."。(注:这里的“内部”一词是指被{{with pipeline}}...{{end}}包围起来的部分,即T1所在位置)
    用法2:
    {{with pipeline}} T1 {{else}} T0 {{end}}
    如果通道的值为空,"."不受影响并且执行T0,否则,将通道的值赋给"."并且执行T1。

    说明:{{end}}标签是if、with、range的结束标签。

    【例子:输出字符窜】
    {{""output""}}
    输出一个字符窜常量。

    {{"output"}}
    输出一个原始字符串常量

    {{printf "%q" "output"}}
    函数调用.(等同于:printf("%q", "output")。)

    {{"output" | printf "%q"}}
    竖线"|"左边的结果作为函数最后一个参数。(等同于:printf("%q", "output")。)

    {{printf "%q" (print "out" "put")}}
    圆括号中表达式的整体结果作为printf函数的参数。(等同于:printf("%q", print("out", "put"))。)

    {{"put" | printf "%s%s" "out" | printf "%q"}}
    一个更复杂的调用。(等同于:printf("%q", printf("%s%s", "out", "put"))。)

    {{"output" | printf "%s" | printf "%q"}}
    等同于:printf("%q", printf("%s", "output"))。

    {{with "output"}}{{printf "%q" .}}{{end}}
    一个使用点号"."的with操作。(等同于:printf("%q", "output")。)

    {{with x := "output" | printf "%q"}}{{x}}{{end}}
    with结构,定义变量,值为执行通道函数之后的结果(等同于:$x := printf("%q", "output")。)

    {{with x := "output"}}{{printf "%q"x}}{{end}}
    with结构中,在其它动作中使用定义的变量

    {{with x := "output"}}{{x | printf "%q"}}{{end}}
    同上,但使用了通道。(等同于:printf("%q", "output")。)

    本书《Go语言四十二章经》内容在github上同步地址:https://github.com/ffhelicopter/Go42
    本书《Go语言四十二章经》内容在简书同步地址: https://www.jianshu.com/nb/29056963

    虽然本书中例子都经过实际运行,但难免出现错误和不足之处,烦请您指出;如有建议也欢迎交流。
    联系邮箱:roteman@163.com

    相关文章

      网友评论

          本文标题:《Go语言四十二章经》第三十五章 模板

          本文链接:https://www.haomeiwen.com/subject/eznmnftx.html