卓越飞翔博客卓越飞翔博客

卓越飞翔 - 您值得收藏的技术分享站
技术文章34734本站已运行393

如何使用 go-yaml 将整数值编组为十六进制?

如何使用 go-yaml 将整数值编组为十六进制?

php小编香蕉为您介绍如何使用 go-yaml 将整数值编组为十六进制。go-yaml 是一个用于处理 YAML 格式数据的 Go 语言库,它提供了简单易用的 API。要将整数值编组为十六进制,我们可以先将整数转换为字节切片,然后使用 fmt.Sprintf 函数将字节切片格式化为十六进制字符串。最后,我们可以使用 go-yaml 库将格式化后的字符串写入 YAML 文件中。这种方法简单高效,能够满足我们的需求。接下来,让我们详细了解一下具体的实现步骤。

问题内容

我有一个带有整数字段的结构,其以十六进制形式表示对人类有意义。例如,将其设为供应商 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
}
  • 分叉 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,并且不需要您维护分支。

    卓越飞翔博客
    上一篇: 删除元素方法在 Golang 中不起作用
    下一篇: 返回列表
    留言与评论(共有 0 条评论)
       
    验证码:
    隐藏边栏