登录
首页 >  Golang >  Go问答

使用 go-yaml 将整数值转换为十六进制编码的方法指南

来源:stackoverflow

时间:2024-02-06 19:30:23 239浏览 收藏

Golang小白一枚,正在不断学习积累知识,现将学习到的知识记录一下,也是将我的所得分享给大家!而今天这篇文章《使用 go-yaml 将整数值转换为十六进制编码的方法指南》带大家来了解一下##content_title##,希望对大家的知识积累有所帮助,从而弥补自己的不足,助力实战开发!


问题内容

我有一个带有整数字段的结构,其以十六进制形式表示对人类有意义。例如,将其设为供应商 ID 字段。

我想将此数据保存到 YAML 文件中以进行手动编辑,然后从文件中加载。据我了解,YAML本身中数字的十六进制表示没有问题,但是 go-yaml (我使用 v3 )以十进制形式编码整数,我还没有找到一种正常的方法来让它以十六进制形式保存它们。

让我们以以下代码为起点:

import (
    //...
    "gopkg.in/yaml.v3"
)

type DeviceInfo struct {
    VendorId uint32 `yaml:"vendorid"`
}

func main() {
    deviceInfo := DeviceInfo{VendorId: 0xdeadbeef}

    yml, err := yaml.Marshal(deviceInfo)
    if err != nil {
        log.Fatal(err)
    }

    fmt.Println(string(yml))
}

此代码生成带有十进制值的 YAML:

vendorid: 3735928559

接下来,go-yaml 允许您为自己的类型创建自定义封送拆收器。我这样做了(我故意省略了 fmt.Sprintf() 格式字符串中的 0x 前缀):

type Uint32Hex uint32

func (U Uint32Hex) MarshalYAML() (interface{}, error) {
    return fmt.Sprintf("%x", U), nil
}

type DeviceInfo struct {
    VendorId Uint32Hex `yaml:"vendorid"`
}

func main() {
    // the same code
}

此代码生成十六进制值,但没有 0x 前缀(目前合乎逻辑):

vendorid: deadbeef

但是如果我在自定义封送拆收器中添加 0x 前缀:

func (U Uint32Hex) MarshalYAML() (interface{}, error) {
    return fmt.Sprintf("0x%x", U), nil
}

该值已正确生成,但不是数字,而是字符串:

vendorid: "0xdeadbeef"

为了将这一行解组为数字,我还必须编写一个自定义解组器。我不喜欢这个解决方案。

最后我有以下几个问题:

  1. 有没有我没有找到的使用 go-yaml 生成十六进制数字的简单方法?

  2. 是否可以根据 go-yaml 作为软件包的扩展来制作自定义编码器,而不更改软件包本身?对我来说,更方便的方法是在结构描述中传递格式标记,例如,如下所示:

    type DeviceInfo struct {
        VendorId uint32 `yaml:"vendorid,hex"`
    }
    
  3. 如果这需要更改包代码,对于这种情况,Go 的做法是什么?只需将包文件复制到我的项目中,根据需要修改并导入本地即可?


正确答案


这里的问题是,在 yaml 中引用字符串是可选的,但 go-yaml 使用与 JSON 编码器相同的内部架构。这意味着首先处理自定义封送处理,然后完全独立地应用引用逻辑。 deadbeef 没有被引用,但 0xdeadbeef 被引用的原因是因为后者是一个数字。它被引用,这样当它知道它应该是一个字符串时,它就不会意外地被解组为数字,因为您的自定义封送拆收器返回了一个字符串。由于 deadbeef 无法被读取为有效数字,因此不需要加引号。您可以做两件事:

  1. 从字符串完成自定义编组/解组:
func (U *Uint32Hex) UnmarshalYAML(value *yaml.Node) error {
    parsed, err := strconv.ParseUint(value.Value, 0, 32)
    *U = Uint32Hex(parsed)
    return err
}
  1. 分叉 go-yaml 并修改它。如果您这样做,则不应更改源文件中的导入。相反,您应该将 replace 指令添加到 go.mod,如下所示:
require gopkg.in/yaml.v3 v3.0.1

replace gopkg.in/yaml.v3 v3.0.1 => ../yaml.v3 // Your local path to the fork

我更喜欢解决方案 1,因为它允许您以自己喜欢的方式序列化这些值,而不会偏离其他人生成的 yaml,并且不需要您维护分支。

理论要掌握,实操不能落!以上关于《使用 go-yaml 将整数值转换为十六进制编码的方法指南》的详细介绍,大家都掌握了吧!如果想要继续提升自己的能力,那么就来关注golang学习网公众号吧!

声明:本文转载于:stackoverflow 如有侵犯,请联系study_golang@163.com删除
相关阅读
更多>
最新阅读
更多>
课程推荐
更多>