我们的 idea 是为GO模板定义一个变量,该模板也是一个使用变量(嵌套模板)的模板,如下所示:

package main

import (
    "os"
    "text/template"
)

type Todo struct {
    Name        string
    Description string
    Subtemplate string
}

func main() {
    td := Todo{
        Name: "Test name",
        Description: "Test description",
        Subtemplate: "Subtemplate {{.Name}}",
    }

    t, err := template.New("todos").Parse("{{.Subtemplate}} You have a task named \"{{ .Name}}\" with description: \"{{ .Description}}\"")
    if err != nil {
        panic(err)
    }
    err = t.Execute(os.Stdout, td)
    if err != nil {
        panic(err)
    }
}

但是,上述代码的结果是:

Subtemplate {{.Name}} You have a task named "Test name" with description: "Test description"

表示没有解析子模板中的变量.Name(可能在设计上是不可能的,需要某种递归调用).有没有什么/其他方法可以达到这个效果?

它也应该适用于使用template.FuncMap定义的模板函数.谢谢.

推荐答案

您可以注册一个解析字符串模板并执行它的函数.

该函数可能如下所示:

func exec(body string, data any) (string, error) {
    t, err := template.New("").Parse(body)
    if err != nil {
        return "", err
    }
    buf := &strings.Builder{}
    err = t.Execute(buf, data)
    return buf.String(), err
}

您将模板正文文本和模板执行数据传递给它.它执行它并返回结果.

注册后,您可以从模板中这样调用它:

{{exec .Subtemplate .}}

完整示例:

td := Todo{
    Name:        "Test name",
    Description: "Test description",
    Subtemplate: "Subtemplate {{.Name}}",
}

t, err := template.New("todos").Funcs(template.FuncMap{
    "exec": func(body string, data any) (string, error) {
        t, err := template.New("").Parse(body)
        if err != nil {
            return "", err
        }
        buf := &strings.Builder{}
        err = t.Execute(buf, data)
        return buf.String(), err
    },
}).Parse("{{exec .Subtemplate .}} You have a task named \"{{ .Name}}\" with description: \"{{ .Description}}\"")
if err != nil {
    panic(err)
}
err = t.Execute(os.Stdout, td)
if err != nil {
    panic(err)
}

这将输出(在Go Playground上试用):

Subtemplate Test name You have a task named "Test name" with description: "Test description"

请注意,如果子模板在运行时没有更改,您应该预先解析它并存储结果模板(*template.Template),以避免每次执行模板时都必须解析它.

Go相关问答推荐

golang父进程的副本无法进行https/tls调用并获得tls:未能验证证书""

golang 的条件储存库

AWS S3 SelectObjectContent在AWS SDK v2 for Go中不返回结果

困扰围棋官方巡回赛的S建议所有方法都使用同一类型的接收器

在GO中创建[]字符串类型的变量

如何使用 go 读取 RDF xml 文件中的 XML 命名空间属性

我可以扫描表中每个项目的最高范围键值吗?

是否需要手动调用rand.Seed?

AWS Lambda 中的 Websocket URL 超时达到错误

Opensearch 错误 ping 弹性服务器:由未知权威签署的 x509 证书

使用 Go Colly 抓取所有可能的标签并将它们放入一个变量中

如何使用泛型将接口转换为指定类型

如何使用 Go 代理状态为 OK 的预检请求?

Golang invopop jsonschema 使用 if/then/else

使用 Golang SQL 驱动程序连接到snowflake

GRPC 反向代理混淆 GRPC 和 GRPC-Web

使用 bolthold 3 条件进行 boltDB 查询

为什么 go-cmp Equal() 说 struct 不是完全相等的,即使所有字段都非常相等?

有没有办法在golang中映射一组对象?

Go http标准库中的内存泄漏?