结构体编解码神器---Mapstructure

news/2024/10/23 7:22:50/

目录

    • 一:介绍
    • 二:安装
    • 三:使用
      • 3.1: 简单使用案例
      • 3.2: 字段标签
      • 3.3: 结构体嵌套
      • 3.4: 统一存储未映射的值
      • 3.5: 逆向转换
      • 3.6: 收集绑定信息

一:介绍

  • 简言之: mapstructure是GO字典(map[string]interface{})和Go结构体之间转换的编解码工具。
  • 核心方法:mapstructure.Decode(input interface{}, output interface{})

二:安装

  • go get github.com/mitchellh/mapstructure
  • 源码地址:https://github.com/mitchellh/mapstructure
  • 官方文档地址:https://pkg.go.dev/github.com/mitchellh/mapstructure

三:使用

3.1: 简单使用案例

  • 案例:将一个JSON字符串解析为字典对象, 然后把字典绑定到一个结构体对象中。
    package mainimport ("encoding/json""fmt""github.com/mitchellh/mapstructure""log"
    )type Person struct {Name stringAge  intJob  string
    }func main() {// 定义一个JSON字符串dataJson :=`{"name":"renshanwen","age":18,"job": "engineer"}`// 将JOSN字符串解析成字典var m map[string]interface{}err := json.Unmarshal([]byte(dataJson), &m)if err != nil {log.Fatal(err)}// 将结构体m,绑定到结构体对象p中var p Personmapstructure.Decode(m, &p)fmt.Printf("name is %s, age is %d, job is %s.", p.Name, p.Age, p.Job)
    }
    
  • 输出结果:
    name is renshanwen, age is 18, job is engineer.
    

3.2: 字段标签

  • 字段标签作用:定义Go字典和GO结构体字段映射关系。
  • 默认:mapstructure使用结构体中字段的名称(大小写不敏感)做这个映射,
  • 案例:例如我们的结构体有一个Name字段,mapstructure解码时会在map[string]interface{}中查找键名name。
  • 指定关系映射:
    package mainimport ("encoding/json""fmt""github.com/mitchellh/mapstructure""log"
    )type Person struct {Name string `mapstructure:"user_name"`Age  int    `mapstructure:"user_age"`Job  string `mapstructure:"user_job"`
    }func main() {// 定义一个JSON字符串dataJson :=`{"user_name":"renshanwen","user_age":18,"user_job": "engineer"}`// 将JOSN字符串解析成结构体var m map[string]interface{}err := json.Unmarshal([]byte(dataJson), &m)if err != nil {log.Fatal(err)}// 将结构体m,绑定到结构体对象p中var p Personmapstructure.Decode(m, &p)fmt.Printf("name is %s, age is %d, job is %s.", p.Name, p.Age, p.Job)
    }
    

3.3: 结构体嵌套

  • 如果入参JSON结构比较复杂, 可以定义多重结构体, mapstructure会自动帮助我们解析。
    package mainimport ("encoding/json""fmt""github.com/mitchellh/mapstructure""log"
    )type Person struct {Name string `mapstructure:"user_name"`Age  int    `mapstructure:"user_age"`Job  string `mapstructure:"user_job"`
    }// Worker 结构体嵌套
    type Worker struct {Person PersonWork   string `mapstructure:"user_work"`
    }func main() {// 定义一个JSON字符串dataJson := `{"person":{"user_name":"renshanwen","user_age":18,"user_job":"engineer"},"user_work":"code"}`// 将JOSN字符串解析成结构体var m map[string]interface{}err := json.Unmarshal([]byte(dataJson), &m)if err != nil {log.Fatal(err)}// 将结构体m,绑定到结构体对象w中var w Workermapstructure.Decode(m, &w)fmt.Printf("name is %s, age is %d, job is %s ,work is %s.", w.Person.Name, w.Person.Age, w.Person.Job, w.Work)
    }
    

3.4: 统一存储未映射的值

  • 可以定义一个字段,接收全部没有映射的字段。
  • 案例:
    package mainimport ("encoding/json""fmt""github.com/mitchellh/mapstructure""log"
    )type Person struct {Name string `mapstructure:"user_name"`Age  int    `mapstructure:"user_age"`Job  string `mapstructure:"user_job"`
    }// Worker 结构体嵌套
    type Worker struct {Person PersonWork   string                 `mapstructure:"user_work"`Other  map[string]interface{} `mapstructure:",remain"` // 未解析字段默认分配到这里
    }func main() {// 定义一个JSON字符串dataJson := `{"person":{"user_name":"renshanwen","user_age":18,"user_job":"engineer"},"user_work":"code","user_mather": "mather","user_baby": "baby"}`// 将JOSN字符串解析成结构体var m map[string]interface{}err := json.Unmarshal([]byte(dataJson), &m)if err != nil {log.Fatal(err)}// 将结构体m,绑定到结构体对象w中var w Workermapstructure.Decode(m, &w)fmt.Printf("name is %s, age is %d, job is %s ,work is %s other is %s.", w.Person.Name, w.Person.Age, w.Person.Job, w.Work, w.Other)
    }
    
  • 结果:
    name is renshanwen, age is 18, job is engineer ,work is code other is map[user_baby:baby user_mather:mather].
    

3.5: 逆向转换

  • 结构体—>字典
  • 在反向解码时,我们可以为某些字段设置mapstructure:“,omitempty”。这样当这些字段为默认值时,就不会出现在结构的map[string]interface{}。
    package mainimport ("encoding/json""fmt""github.com/mitchellh/mapstructure"
    )type Person struct {Name string `mapstructure:"user_name"`Age  int    `mapstructure:"user_age"`Job  string `mapstructure:"user_job,omitempty"` // ,omitempty表示如果结构体对象没有,则不会解析到字典中。
    }// Worker 结构体嵌套
    type Worker struct {Person PersonWork   string                 `mapstructure:"user_work"`Other  map[string]interface{} `mapstructure:",remain"` // 未解析字段默认分配到这里
    }func main() {// 定义一个JSON字符串worker := &Worker{Work: "code",}worker.Person.Name = "renshanwen"worker.Person.Age = 18var m map[string]interface{}mapstructure.Decode(worker, &m)data, _ := json.Marshal(m)fmt.Println(string(data))// {"Other":null,"Person":{"user_age":18,"user_name":"renshanwen"},"user_work":"code"}
    }
    

3.6: 收集绑定信息

  • 可以使用mapstructure.Metadata, 存储绑定信息情况。
  • Keys: 绑定成功的
  • Unused:没有绑定上的
  • Unset:缺失的
    package mainimport ("fmt""github.com/mitchellh/mapstructure"
    )type Person struct {Name string `mapstructure:"user_name"`Age  int    `mapstructure:"user_age"`Job  string `mapstructure:"user_job,omitempty"` // ,omitempty表示如果结构体对象没有,则不会解析到字典中。
    }func main() {m := map[string]interface{}{"user_name": "renshanwen","age":       18,"job":       "engineer",}var p Personvar metadata mapstructure.Metadatamapstructure.DecodeMetadata(m, &p, &metadata)fmt.Printf("keys:%#v unused:%#v Unset:%#v \n", metadata.Keys, metadata.Unused, metadata.Unset)// 	keys:[]string{"user_name"} unused:[]string{"age", "job"} Unset:[]string{"user_age", "user_job"} 
    }
    

http://www.ppmy.cn/news/1114666.html

相关文章

哪些测试项目可以使用自动化测试?

通常,软件测试v的测试方式分为人工测试和自动化测试,人工测试是由测试人员编写并执行测试用例,然后观察测试结果与预期结果是否一致的过程;自动化测试是通过测试工具来代替或辅助人工去验证系统功能是否有问题的过程。 采用自动化测试需要满…

早知道这些免费 API,我就可以不用到处爬数据了

豆瓣开放 https://developers.douban.com/wiki/ 淘宝开放平台API 商品详情接口 http://open.taobao.com/ IP地址查询 http://apistore.baidu.com/apiworks/servicedetail/114.html 频道新闻API_易源 http://apistore.baidu.com/apiworks/servicedetail/688.html 微信热…

软考 - 系统架构设计师如何备考?文中含资料分享和备考心得

前言 我参加了2022年11月份的《软考-系统架构设计师》考试,在两个多月的备考之中我总结了一些学习经验和答题技巧,现毫无保留的分享给大家,希望对报考的同学们有所帮助。 一、软考的作用 1、以考代评(国企、事业单位、公务员评职…

【Web_环境搭建_Python3_pip】pip的升级、安装、更新、卸载,以及pipupgrade和pip-review的基础使用

** 官方说明 ** pip(Python Package Index)是一个以 Python 语言写成的软件包管理系統,使用 pip 可以非常方便的安装和管理 python 软件包PIP ** 查看信息 ** 查看版本 : pip --version查看已有 : pip list、pip freeze查看帮助 : pip help查看库信息 : pip show -f package_…

多台服务器sessionId共享

目录 多台服务器sessionId共享解决方案:ASP.NET Core 参考代码(NET 7):登录处理登录(请求)过滤器过滤器使用BaseController 多台服务器sessionId共享 session id是服务器首次与浏览器创建连接时,生成的id值,存入浏览器…

【前端】jquery获取data-*的属性值

通过jquery获取下面data-id的值 <div id"getId" data-id"122" >获取id</div> 方法一&#xff1a;dataset()方法 //data-前缀属性可以在JS中通过dataset取值&#xff0c;更加方便 console.log(getId.dataset.id);//112//赋值 getId.dataset.…

扔掉你的开发板,跟我玩Mcore-全志h616

本文转载自WhyCan Forum(哇酷开发者社区)&#xff1a; https://whycan.com/t_10024.html 作者leefei 这是一个1.69寸触摸小电视。使用全志H616芯片&#xff0c;板上硬件有mpu6050陀螺仪&#xff0c;USB转ttl调试串口&#xff0c;一个USB接口&#xff0c;WIFI&蓝牙&#x…

【蓝桥杯选拔赛真题61】Scratch小猫照镜子 少儿编程scratch图形化编程 蓝桥杯选拔赛真题解析

目录 scratch小猫照镜子 一、题目要求 编程实现 二、案例分析 1、角色分析