golang的xml、json解析

xml

golang的xml处理主要应用Unmarshal、Marshal方法实现,解析一个xml到struct如下,首先是xml文件:

<?xml version="1.0" encoding="utf-8"?>
<servers version="1.0">
    <!--测试注释server-->
    <server desc="s1">
        <serverName>Shanghai_VPN</serverName>
        <serverIP>127.0.0.1</serverIP>
    </server>
    <server desc="s2">
        <!---->
        <serverName>Beijing_VPN</serverName>
        <serverIP>127.0.0.2</serverIP>
    </server>
</servers>

下面记录下主要代码:

// parse2xml project main.go
package main

import (
    "encoding/xml"
    "fmt"
    "io/ioutil"
    "os"
)

type Recurlyservers struct {
    XMLName     xml.Name `xml:"servers"`
    Version     string   `xml:"version,attr"`
    Svs         []server `xml:"server"`
    Description string   `xml:",innerxml"`
}

type server struct {
    XMLName    xml.Name `xml:"server"`
    Desc       string   `xml:"desc,attr"`
    ServerName string   `xml:"serverName"`
    ServerIP   string   `xml:"serverIP"`
    TestDesc   string   `xml:",innerxml"`
}

func main() {
    
    file, err := os.Open("servers.xml")
    if err != nil {
        fmt.Printf("%s", err.Error())
        panic(err)
    }
    defer file.Close()
    data, err := ioutil.ReadAll(file)
    v := Recurlyservers{}
    err = xml.Unmarshal(data, &v)
    if err != nil {
        fmt.Printf("%s", err.Error())
        panic(err)
    }
    fmt.Printf("%#v", v)
}

打印结果:

可以发现Unmarshal解析时的一些规则:

1、解析使用struct的tag配置,通过底层反射实现

2、类型为xml.Name的struct字段XMLName对应的是xml中的“父”节点名称,如servers

3、*,attr对应的为当前父节点上的属性名称,如version,attr,对应的便是xml文件中servers节点的version属性

4、tag中仅写一个名称如serverName,表示serverName为节点名称

5、对于某个struct中,innerxml的tag标注,该struct的属性对应的数据为该struct相应的XMLName对应的xml下的所有内容,比如TestDesc对应的为server节点下的所有内容,Description对应的便是servers下的所有内容

由struct生成对应的xml,方法如下:

// struct2xmll project main.go
package main

import (
    "encoding/xml"
    "fmt"
    "os"
)

type Servers struct {
    XMLName xml.Name `xml:"servers"`
    Version string   `xml:"version,attr"`
    Svs     []server `xml:"server"`
}

type server struct {
    ServerName string `xml:"serverName"`
    ServerIP   string `xml:"serverIP"`
}

func main() {
    v := &Servers{Version: "1.0"}
    v.Svs = append(v.Svs, server{"Shanghai_VPN", "127.0.0.1"})
    v.Svs = append(v.Svs, server{"Beijing_VPN", "127.0.0.1"})
    output, err := xml.MarshalIndent(v, " ", " ")
    if err != nil {
        fmt.Println(err.Error())
        return
    }
    os.Stdout.Write([]byte(xml.Header))
    file, err := os.Create("servers.xml")
    file.Write([]byte(xml.Header))
    file.Write(output)
}

生成的xml文档:

json

将json字符串解析为struct:

// jsonparse project main.go
package main

import (
    "encoding/json"
    "fmt"
)

type Server struct {
    ServerName string `json:"serverName"`
    ServerIP   string `json:"serverIP"`
}

type Serversslice struct {
    Servers []Server `json:"servers"`
}

func main() {
    var s Serversslice
    str := `{"servers":[{"serverName":"Shanghai_VPN","serverIP":"127.0.0.1"},{"serverName":"Beijing_VPN","serverIP":"192.168.20.132"}]}`
    json.Unmarshal([]byte(str), &s)
    fmt.Printf("%#v", s)
}

结果:

将一个struct解析为json如下:

package main

import (
    "encoding/json"
    "fmt"
)

type Server struct {
    serverName string `json:"serverName"`
    ServerIP   string `json:"serverIP"`
}

type ServersSlice struct {
    Servers []Server `json:"servers"`
    Desc    string   `json:"desc"`
}

func main() {
    servers := []Server{{serverName: "admin", ServerIP: "192.168.20.131"}, {serverName: "admin1", ServerIP: "192.168.20.132"}}
    desc := "描述部分"
    var ss ServersSlice
    ss.Desc = desc
    ss.Servers = servers
    js, _ := json.Marshal(ss)
    fmt.Printf("Json: %s", js)
}

结果:

本文参与腾讯云自媒体分享计划,欢迎正在阅读的你也加入,一起分享。

发表于

我来说两句

0 条评论
登录 后参与评论

相关文章

来自专栏白驹过隙

ACE - 代码层次及Socket封装

原文出自http://www.cnblogs.com/binchen-china,禁止转载。

551
来自专栏蓝天

boost库thread.hpp编译警告honored已修复

请浏览:https://svn.boost.org/trac/boost/ticket/7874

872
来自专栏码匠的流水账

bucket4j使用实例

bucket4j类库是一款优秀的java限流类库,可以用来限流,也可以用作简单调度。

291
来自专栏AILearning

安装 virtualenv 和 virtualenvwrapper 来管理环境

安装 virtualenv 和 virtualenvwrapper 来管理环境 安装 python 参考链接: http://www.tuicool.co...

1816
来自专栏互联网杂技

SpringBoot ( 七 ) :springboot + mybatis 多数据源最简解决方案

说起多数据源,一般都来解决那些问题呢,主从模式或者业务比较复杂需要连接不同的分库来支持业务。我们项目是后者的模式,网上找了很多,大都是根据jpa来做多数据源解决...

1103
来自专栏乐沙弥的世界

Oracle 11g RAC 执行root.sh时遭遇 CRS-0184/PRCR-1070

Oracle 11g RAC安装时,在第一个节点执行root.sh时遭遇了CRS-0184/PRCR-1070,Google了很多帖子也没有找到解决办法。呜呜,...

671
来自专栏沃趣科技

ASM 翻译系列第四弹:高级知识 kfed 元数据编辑器

原作者:Bane Radulovic 译者: 赵恩东 审核: 魏兴华 DBGeeK社群联合出品 kfed - ASM metadata editor...

3596
来自专栏不会写文章的程序员不是好厨师

日志那些事儿——slf4j集成logback/log4j

在日志Logger漫谈中提到了slf4j仅仅是作为日志门面,给用户提供统一的API使用,而真正的日志系统的实现是由logback或者log4j这样的日志系统实现...

692
来自专栏沃趣科技

ASM 翻译系列第二十弹:ASM Internal ASM file number 7

原作者:Bane Radulovic 译者: 郭旭瑞 审核: 魏兴华 DBGeeK社群联合出品 ASM file number 7 ASM元信息7号...

3627
来自专栏扎心了老铁

Mybatis分页插件PageHelper的配置和使用方法

前言 在web开发过程中涉及到表格时,例如dataTable,就会产生分页的需求,通常我们将分页方式分为两种:前端分页和后端分页。 前端分页 一次性请求数据表格...

4744

扫码关注云+社区