11

当我尝试访问要传递给模板的函数时出现错误:

Error: template: struct.tpl:3: function "makeGoName" not defined

谁能让我知道我做错了什么?

模板文件(struct.tpl):

type {{.data.tableName}} struct {
  {{range $key, $value := .data.tableData}}
  {{makeGoName $value.colName}} {{$value.colType}} `db:"{{makeDBName $value.dbColName}},json:"{{$value.dbColName}}"`
  {{end}}
}

调用文件:

type tplData struct {
    tableName string
    tableData interface{}
}

func doStuff() {
    t, err := template.ParseFiles("templates/struct.tpl")
    if err != nil {
        errorQuit(err)
    }

    t = t.Funcs(template.FuncMap{
        "makeGoName": makeGoName,
        "makeDBName": makeDBName,
    })

    data := tplData{
        tableName: tableName,
        tableData: tableInfo,
    }

    t.Execute(os.Stdout, data)
}

func makeGoName(name string) string {
    return name
}

func makeDBName(name string) string {
    return name
}

这是一个生成结构样板代码的程序(以防有人想知道我为什么在我的模板中这样做)。

4

2 回答 2

22

自定义函数需要在解析模板之前注册,否则解析器将无法判断标识符是否是有效的函数名。模板被设计为可静态分析的,这是对此的要求。

你可以先用 新建一个未定义的模板template.New(),除了函数,类型(由返回)还有一个方法,你可以调用它。template.ParseFiles() template.TemplateNew()Template.ParseFiles()

像这样的东西:

t, err := template.New("").Funcs(template.FuncMap{
    "makeGoName": makeGoName,
    "makeDBName": makeDBName,
}).ParseFiles("templates/struct.tpl")

请注意,该template.ParseFiles()函数还在后台调用template.New(),将第一个文件的名称作为模板名称传递。

Template.Execute()返回一个error, 打印以查看是否没有生成输出,例如:

if err := t.Execute(os.Stdout, data); err != nil {
    fmt.Println(err)
}
于 2016-02-22T10:03:43.303 回答
1

在为您的模板注册自定义函数并使用ParseFiles()时,您需要在实例化和执行模板时指定模板的名称。您还需要在调用ParseFiles()后调用Funcs()

// Create a named template with custom functions
t, err := template.New("struct.tpl").Funcs(template.FuncMap{
    "makeGoName": makeGoName,
    "makeDBName": makeDBName,
}).ParseFiles("templates/struct.tpl") // Parse the template file
if err != nil {
    errorQuit(err)
}

// Execute the named template
err = t.ExecuteTemplate(os.Stdout, "struct.tpl", data)
if err != nil {
    errorQuit(err)
}

使用命名模板时,名称是没有目录路径的文件名,例如struct.tplnot templates/struct.tpl。所以 nameNew()应该ExecuteTemplate()是 string struct.tpl

于 2020-08-22T23:37:43.323 回答