Text/template: "can't call method/function with 0 results."

Text/template: "can't call method/function with 0 results."

如何在 returns 没有价值的模板中执行函数?这是示例:

func main() {
    u, err := url.Parse("http://example.com/test?param1=true&param2=true")
    if err != nil {
        log.Fatal(err)
    }
    m := u.Query()
    m.Del("param1") // param1 successful deleted!
    u.RawQuery = m.Encode()
    fmt.Println(u.RawQuery)

    const tmpl = `
    {{$m := .Query}}
    {{$m.Del "param2"}} <!-- failed to delete param2! -->
    {{.RawQuery}}
    `
    t := template.Must(template.New("").Parse(tmpl))
    err = t.Execute(os.Stdout, u)
    if err != nil {
        log.Println("executing template:", err)
    }

}
see this code in play.golang.org


我知道在模板中不应该有太多逻辑,但是对 运行 函数的无知 returns 没有价值在我看来是个有趣的问题。

Go 中的模板不同于其他语言中的模板(例如 PHP)。使用 template.FuncMap 为您的模板创建自定义函数。

package main

import (
    "fmt"
    "log"
    "net/url"
    "os"
    "text/template"
)

func main() {
    funcMap := template.FuncMap{
        "delete": deleteMap,
    }

    u, err := url.Parse("http://example.com/test?param1=true&param2=true")
    if err != nil {
        log.Fatal(err)
    }

    u = deleteMap(u, "param1") // works in regular code and templates

    fmt.Println(u.RawQuery)

    const tmpl = `
    {{$m := delete . "param2"}} <!-- WORKS! -->
    {{$m.RawQuery}}
    `
    t := template.New("").Funcs(funcMap)
    t = template.Must(t.Parse(tmpl))

    err = t.Execute(os.Stdout, u)

    if err != nil {
        log.Println("executing template:", err)
    }

}

func deleteMap(u *url.URL, key string) *url.URL {
    m := u.Query()
    m.Del(key) // key successful deleted!
    u.RawQuery = m.Encode()
    return u
}

或者,尝试 playground version