23

这个看起来很简单,但它让我发疯了。

如何在 golang 模板的嵌套范围内引用范围内更高的结构元素?

例子:

type Foo struct {
  Id string
  Name string
}

type Bar struct {
  Id string
  Name string
}

var foos []Foo
var bars []Bar

// logic to populate both foos and bars

在模板中:

{{range .foos}}
  <div>Foo {{.Name}}</div>
  <div>
    {{range ..bars}}
      <div>Bar {{.Name}} <input type="text" name="ids_{{..Id}}_{{.Id}}" /></div>
    {{end}}
  </div>
{{end}}

显然 ..bars 和 ..Id 不起作用,但希望我的意图很明确。我想遍历 Foo 和 Bar 的所有组合,并生成一个表单元素,其名称由 Foo 的 Id 和 Bar 的 Id 构建。

问题是似乎不可能:

  1. 从 foos 范围范围内访问栏
  2. 从栏的范围范围内访问 Foo 的 Id

我通过在两个结构中放置一堆冗余字段来临时解决这个问题,但这对我来说似乎很丑陋,违反了 DRY,而且总体上感觉非常错误。

golang模板有什么办法可以做我想做的事吗?

4

1 回答 1

38

是的。我觉得好像没有找到解决方案是因为没有text/template足够仔细地阅读包。如果您使用html/template的是 ,则语法是相同的(它们会告诉您阅读文本/模板;))。这是您可能想要做的完整工作解决方案。

去文件:

package main

import (
    "bytes"
    "io/ioutil"
    "os"
    "strconv"
    "text/template"
)

type Foo struct {
    Id   string
    Name string
}

type Bar struct {
    Id   string
    Name string
}

var foos []Foo
var bars []Bar

func main() {
    foos = make([]Foo, 10)
    bars = make([]Bar, 10)

    for i := 0; i < 10; i++ {
        foos[i] = Foo{strconv.Itoa(i), strconv.Itoa(i)} // just random strings
        bars[i] = Bar{strconv.Itoa(10 * i), strconv.Itoa(10 * i)}
    }

    tmpl, err := ioutil.ReadFile("so.tmpl")
    if err != nil {
        panic(err)
    }

    buffer := bytes.NewBuffer(make([]byte, 0, len(tmpl)))

    output := template.Must(template.New("FUBAR").Parse(string(tmpl)))
    output.Execute(buffer, struct {
        FooSlice []Foo
        BarSlice []Bar
    }{
        FooSlice: foos,
        BarSlice: bars,
    })

    outfile, err := os.Create("output.html")
    if err != nil {
        panic(err)
    }
    defer outfile.Close()
    outfile.Write(buffer.Bytes())
}

注意:您可能可以做一些事情来不将文件加载到中间缓冲区(使用ParseFiles),我只是复制并粘贴了一些我为我的一个项目编写的代码。

模板文件:

{{ $foos := .FooSlice }}
{{ $bars := .BarSlice }}

{{range $foo := $foos }}
  <div>Foo {{$foo.Name}}</div>
  <div>
    {{range $bar := $bars}}
      <div>Bar {{$bar.Name}} <input type="text" name="ids_{{$foo.Id}}_{{$bar.Id}}" /></div>
    {{end}}
  </div>
{{end}}

这个故事的两个寓意是
a) 明智地使用模板中的变量,它们是有益的
b) 模板中的范围也可以设置变量,您不需要单独依赖$.

于 2013-07-07T05:43:28.183 回答