当前位置:首页 > 文章列表 > Golang > Go问答 > 调用 BindPFlag 时为什么会收到 nil 指针错误?

调用 BindPFlag 时为什么会收到 nil 指针错误?

来源:stackoverflow 2024-03-25 09:18:36 0浏览 收藏

在使用 Cobra 和 Viper 库时,调用 `config.BindPFlag` 时可能会收到 `nil` 指针错误。原因是 `cmd.flags().lookup("name")` 返回 `nil`,因为持久标志尚未在 `cmd.newcmdroot()` 级别初始化。要解决此问题,需要使用 `cmd.PersistentFlags().Lookup("name")` 来查找持久标志,而不是 `cmd.flags().lookup("name")`。

问题内容

我最近刚刚开始使用 go,并且遇到了一些问题 我不确定我是否理解与 cobra 和 viper 一起工作的行为。

这是您获得的示例代码的稍微修改版本 运行 cobra init。在 main.go 我有:

package main

import (
    "github.com/larsks/example/cmd"
    "github.com/spf13/cobra"
)

func main() {
    rootcmd := cmd.newcmdroot()
    cobra.checkerr(rootcmd.execute())
}

cmd/root.go 中我有:

package cmd

import (
    "fmt"
    "os"

    "github.com/spf13/cobra"

    "github.com/spf13/viper"
)

var cfgfile string

func newcmdroot() *cobra.command {
    config := viper.new()

    var cmd = &cobra.command{
        use:   "example",
        short: "a brief description of your application",
        persistentprerun: func(cmd *cobra.command, args []string) {
            initconfig(cmd, config)
        },
        run: func(cmd *cobra.command, args []string) {
            fmt.printf("this is a test\n")
        },
    }

    cmd.persistentflags().stringvar(&cfgfile, "config", "", "config file (default is $home/.example.yaml)")
    cmd.persistentflags().string("name", "", "a name")

  // *** if i move this to the top of initconfig
  // *** the code runs correctly.
    config.bindpflag("name", cmd.flags().lookup("name"))

    return cmd
}

func initconfig(cmd *cobra.command, config *viper.viper) {
    if cfgfile != "" {
        // use config file from the flag.
        config.setconfigfile(cfgfile)
    } else {
        config.addconfigpath(".")
        config.setconfigname(".example")
    }

    config.automaticenv() // read in environment variables that match

    // if a config file is found, read it in.
    if err := config.readinconfig(); err == nil {
        fmt.fprintln(os.stderr, "using config file:", config.configfileused())
    }

  // *** this line triggers a nil pointer reference.
    fmt.printf("name is %s\n", config.getstring("name"))
}

此代码将在最终调用时因 nil 指针引用而崩溃 fmt.printf

panic: runtime error: invalid memory address or nil pointer dereference
[signal SIGSEGV: segmentation violation code=0x1 addr=0x50 pc=0x6a90e5]

如果我将调用从 newcmdroot 移至 config.bindpflag 函数到 initconfig 命令的顶部,一切都运行 没有问题。

这是怎么回事?根据 viper 文档有关使用 bindpflags

与 bindenv 一样,绑定方法时不设置该值 被调用,但是当它被访问时。这意味着您可以尽早绑定 你想要的,即使是在 init() 函数中。

这几乎正是我在这里所做的。当我打电话的时候 config.bindpflagconfig 为非零,cmd 为非零,并且 name 参数已注册。

我认为我在 a 中使用 config 时发生了一些问题 persistentprerun 中的闭包,但我不知道具体原因 导致此失败。


正确答案


如果我使用 cmd.persistentflags().lookup("name"),我不会遇到任何问题。

// *** if i move this to the top of initconfig
    // *** the code runs correctly.
    pflag := cmd.persistentflags().lookup("name")
    config.bindpflag("name", pflag)

考虑到您刚刚注册了 persistent flags(该标志将适用于它分配给的命令以及该命令下的每个命令),调用 cmd.persistentflags().lookup("name") 更安全,而不是cmd.flags().lookup("名称")

后者返回 nil,因为只有在调用 rootcmd.execute() 时才会调用 persistentprerun,即在 cmd.newcmdroot() 之后
cmd.newcmdroot() 级别,标志尚未初始化,即使在某些标志被声明为“持久”之后也是如此。

我认为这很有趣,所以我做了一些挖掘和 found your exact problem documented in an issue。有问题的行是这样的:

config.bindpflag("name", cmd.flags().lookup("name"))
//                           ^^^^^^^

您创建了一个持久标志,但将该标志绑定到 flags 属性。如果您更改代码以绑定到 persistentflags,即使使用 newcmdroot 中的这一行,一切也会按预期工作:

config.BindPFlag("name", cmd.PersistentFlags().Lookup("name"))

理论要掌握,实操不能落!以上关于《调用 BindPFlag 时为什么会收到 nil 指针错误?》的详细介绍,大家都掌握了吧!如果想要继续提升自己的能力,那么就来关注golang学习网公众号吧!

版本声明
本文转载于:stackoverflow 如有侵犯,请联系study_golang@163.com删除
Golang socket.io 服务器遭遇故障的原因Golang socket.io 服务器遭遇故障的原因
上一篇
Golang socket.io 服务器遭遇故障的原因
设置MQTT连接的超时时间的有效方法
下一篇
设置MQTT连接的超时时间的有效方法
查看更多
最新文章
查看更多
课程推荐
  • 前端进阶之JavaScript设计模式
    前端进阶之JavaScript设计模式
    设计模式是开发人员在软件开发过程中面临一般问题时的解决方案,代表了最佳的实践。本课程的主打内容包括JS常见设计模式以及具体应用场景,打造一站式知识长龙服务,适合有JS基础的同学学习。
    542次学习
  • GO语言核心编程课程
    GO语言核心编程课程
    本课程采用真实案例,全面具体可落地,从理论到实践,一步一步将GO核心编程技术、编程思想、底层实现融会贯通,使学习者贴近时代脉搏,做IT互联网时代的弄潮儿。
    511次学习
  • 简单聊聊mysql8与网络通信
    简单聊聊mysql8与网络通信
    如有问题加微信:Le-studyg;在课程中,我们将首先介绍MySQL8的新特性,包括性能优化、安全增强、新数据类型等,帮助学生快速熟悉MySQL8的最新功能。接着,我们将深入解析MySQL的网络通信机制,包括协议、连接管理、数据传输等,让
    498次学习
  • JavaScript正则表达式基础与实战
    JavaScript正则表达式基础与实战
    在任何一门编程语言中,正则表达式,都是一项重要的知识,它提供了高效的字符串匹配与捕获机制,可以极大的简化程序设计。
    487次学习
  • 从零制作响应式网站—Grid布局
    从零制作响应式网站—Grid布局
    本系列教程将展示从零制作一个假想的网络科技公司官网,分为导航,轮播,关于我们,成功案例,服务流程,团队介绍,数据部分,公司动态,底部信息等内容区块。网站整体采用CSSGrid布局,支持响应式,有流畅过渡和展现动画。
    484次学习
查看更多
AI推荐
  • 千音漫语:智能声音创作助手,AI配音、音视频翻译一站搞定!
    千音漫语
    千音漫语,北京熠声科技倾力打造的智能声音创作助手,提供AI配音、音视频翻译、语音识别、声音克隆等强大功能,助力有声书制作、视频创作、教育培训等领域,官网:https://qianyin123.com
    170次使用
  • MiniWork:智能高效AI工具平台,一站式工作学习效率解决方案
    MiniWork
    MiniWork是一款智能高效的AI工具平台,专为提升工作与学习效率而设计。整合文本处理、图像生成、营销策划及运营管理等多元AI工具,提供精准智能解决方案,让复杂工作简单高效。
    170次使用
  • NoCode (nocode.cn):零代码构建应用、网站、管理系统,降低开发门槛
    NoCode
    NoCode (nocode.cn)是领先的无代码开发平台,通过拖放、AI对话等简单操作,助您快速创建各类应用、网站与管理系统。无需编程知识,轻松实现个人生活、商业经营、企业管理多场景需求,大幅降低开发门槛,高效低成本。
    172次使用
  • 达医智影:阿里巴巴达摩院医疗AI影像早筛平台,CT一扫多筛癌症急慢病
    达医智影
    达医智影,阿里巴巴达摩院医疗AI创新力作。全球率先利用平扫CT实现“一扫多筛”,仅一次CT扫描即可高效识别多种癌症、急症及慢病,为疾病早期发现提供智能、精准的AI影像早筛解决方案。
    179次使用
  • 智慧芽Eureka:更懂技术创新的AI Agent平台,助力研发效率飞跃
    智慧芽Eureka
    智慧芽Eureka,专为技术创新打造的AI Agent平台。深度理解专利、研发、生物医药、材料、科创等复杂场景,通过专家级AI Agent精准执行任务,智能化工作流解放70%生产力,让您专注核心创新。
    192次使用
微信登录更方便
  • 密码登录
  • 注册账号
登录即同意 用户协议隐私政策
返回登录
  • 重置密码