当前位置: 首页 > 面试题库 >

如何在golang html / template中创建全局变量并在多个位置进行更改?

伯逸明
2023-03-14
问题内容

我在中创建一个变量,html/template并根据条件更改值。但是值的范围仅保留在以下if条件内:

{{if .UserData}}
    {{$currentUserId := .UserData.UserId}}
    [<a href="#ask_question">Inside {{$currentUserId}}</a>]
{{else}}
    {{$currentUserId := 0}}
{{end}}
[<a href="#ask_question">outside {{$currentUserId}}</a>]

在if条件内,我得到正确的值,但在外面0。如何在$currentUserId外界条件下使用?有人可以帮我吗?


问题答案:

Go
1.11添加了对更改模板变量值的支持。要定义变量,请使用:=

{{$currentUserId := 0}}

要更改其值,请使用Assignment =

{{$currentUserId = .UserData.UserId}}

如果变量在{{if}}块外创建但在块内更改,则更改将在{{if}}块后可见。

{{$currentUserId := 0 -}}
Before: {{$currentUserId}}
{{if .UserData -}}
    {{$currentUserId = .UserData.UserId}}
    [<a href="#ask_question">Inside {{$currentUserId}}</a>]
{{else}}
    {{$currentUserId = 0}}
{{end}}
[<a href="#ask_question">outside {{$currentUserId}}</a>]

像这样测试:

m := map[string]interface{}{}
t := template.Must(template.New("").Parse(src))

m["UserData"] = UserData{99}
if err := t.Execute(os.Stdout, m); err != nil {
    panic(err)

输出为(在Go Playground上尝试):

Before: 0

    [<a href="#ask_question">Inside 99</a>]

[<a href="#ask_question">outside 99</a>]

原始答案如下。

简短的答案是:您不能。

根据设计原理,模板不应包含复杂的逻辑。在模板中,您只能创建新变量,而不能更改现有模板变量的值。当您{{$currentUserId := .UserData.UserId}}在一个{{if}}块中执行操作时,这将创建一个新变量,该变量将外部变量遮盖起来,并且其作用域扩展到{{end}}动作。

text/template包的“
变量”部分对此进行了描述(html/template包也是如此):

变量的范围扩展到声明它的控制结构(“ if”,“ with”或“
range”)的“结束”动作,或者扩展到模板的末尾(如果没有这样的控制结构)。模板调用不会从调用点继承变量。

可能的解决方法

在您的情况下,最简单的方法是将自定义函数注册CurrentUserId()到您的模板,如果UserData存在,则返回其id
UserData.UserId(),如果不存在,则按0您的情况返回。

这是一个如何完成的示例:

type UserData struct {
    UserId int
}

func main() {
    m := map[string]interface{}{}
    t := template.Must(template.New("").Funcs(template.FuncMap{
        "CurrentUserId": func() int {
            if u, ok := m["UserData"]; ok {
                return u.(UserData).UserId
            }
            return 0
        },
    }).Parse(src))

    if err := t.Execute(os.Stdout, m); err != nil {
        panic(err)
    }
    m["UserData"] = UserData{99}
    if err := t.Execute(os.Stdout, m); err != nil {
        panic(err)
    }
}

const src = `Current user id: {{CurrentUserId}}
`

它首先在没有的情况下执行模板UserData,然后在UserData具有的情况下执行模板UserId = 99。输出为:

Current user id: 0
Current user id: 99

在Go Playground上尝试一下。

模拟可变变量

您还可以模拟可变变量,也可以使用注册的自定义函数进行模拟。您可以注册一个像SetCurrentUserId()这样的函数,该函数将更改变量的值,最好是在传递给模板执行的参数中,这样可以安全地并发使用。

这是一个示例(它使用a map作为模板数据,并将SetCurrentUserId()当前用户ID设置为此映射中的值):

func main() {
    m := map[string]interface{}{}
    t := template.Must(template.New("").Funcs(template.FuncMap{
        "SetCurrentUserId": func(id int) string {
            m["CurrentUserId"] = id
            return ""
        },
    }).Parse(src))

    if err := t.Execute(os.Stdout, m); err != nil {
        panic(err)
    }
    m["UserData"] = UserData{99}
    if err := t.Execute(os.Stdout, m); err != nil {
        panic(err)
    }
}

const src = `Before: {{.CurrentUserId}}
{{if .UserData}}
    {{SetCurrentUserId .UserData.UserId}}Inside: {{.CurrentUserId}}
{{else}}
    {{SetCurrentUserId 0}}Inside: {{.CurrentUserId}}
{{end}}
After: {{.CurrentUserId}}
`

这再次首先执行不带模板的模板UserData,然后再使用UserData具有的模板UserId = 99。输出为:

Before: 
    Inside: 0
After: 0
Before: 0
    Inside: 99
After: 99

在Go Playground上尝试一下。



 类似资料:
  • 问题内容: 在我的android应用程序中,我需要放置可变成员id的位置。问题是,它是从在线API获取的,我需要找到一种存储/检索它的方法 我尝试将其放在自定义类中,但是问题是,如果我取消活动,它将丢失,我还知道有一种方法可以扩展应用程序。 所以我想知道存储全局变量的最佳方法是什么? 我必须实现: 将变量保存在onSaveState上 将其保存在sharepref 手动保存 手动检索 谢谢 更新:

  • 问题内容: 我试图设置一个全局变量。就我而言,只是一个布尔标志,用于指示是否首次显示视图: 呈现视图后,我想将此标志设置为false: 然后检查它: 因此,我想创建一个全局变量。在哪里以及如何?我试过了: 在我的视图控制器领域 在我的AppDelegate.swift文件中的方法中 在班上 没运气。我收到一条错误消息 (注意:我意识到,在这个问题中,我背叛了我对Swift中如何处理范围的无知。请原

  • 在JavaScript中,我可以这样做: 然后在另一个文件中: 它将定义

  • 问题内容: 我有一个全局变量,需要在我的ViewController之间共享。 在Objective-C中,我可以定义静态变量,但是找不到在Swift中定义全局变量的方法。 您知道这样做的方法吗? 问题答案: 来自官方的Swift编程指南: 全局变量是在任何函数,方法,闭包或类型上下文之外定义的变量。全局常数和变量总是延迟计算的。 您可以在任何文件中定义它,也可以在任何位置访问它。因此,您可以在任

  • 本文向大家介绍如何在函数中设置一个全局变量相关面试题,主要包含被问及如何在函数中设置一个全局变量时的应答技巧和注意事项,需要的朋友参考一下 python中的global语句是被用来声明全局变量的。

  • 问题内容: 我正在一个项目上,我想让我的索引在滚动时加载更多内容。我想设置一个全局变量来保存页面已加载多少次。我的项目结构如下: 首先,我在中声明全局变量: 皮查姆(Pycharm)警告 在: 还有 我也引用了global-variable-and-python-flask 但我没有main()函数。在Flask中设置全局变量的正确方法是什么? 问题答案: 你没有在定义,只是在声明,所以就好像在说