백엔드 개발 Golang Go 언어로 json 데이터를 처리하는 방법

Go 언어로 json 데이터를 처리하는 방법

Jan 11, 2020 pm 05:14 PM
언어로 가다

Go 언어로 json 데이터를 처리하는 방법

go json bag armarshal(): GO 데이터 객체-& gt; json 데이터 Unmarshal(): json 데이터-& gt; Go 데이터 객체

func Marshal(v interface{}) ([]byte, error)
func Unmarshal(data []byte, v interface{}) error
로그인 후 복사

JSON 데이터 구축

Marshal() MarshalIndent() 함수는 데이터를 json 데이터로 캡슐화할 수 있습니다. 1. struct, Slice, array, map은 모두 json으로 변환 가능2. struct를 json으로 변환할 때는 필드의 첫 글자만 변환됩니다

3. map 변환 시 키는 문자열이어야 합니다.

4. 캡슐화할 때 포인터인 경우 포인터가 가리키는 개체가 캡슐화를 위해 추적됩니다. 예:

구조체는 다음과 같습니다.

type Post struct {
    Id      int
    Content string
    Author  string
}
로그인 후 복사

이 구조는 기사가 포함된 블로그 게시물 유형을 나타냅니다. ID, 기사 내용, 기사 내용을 제출합니다. 이에 대해 말할 것도 없고 지정해야 할 유일한 것은 구조체이고 구조체가 JSON 데이터로 캡슐화(인코딩)될 수 있다는 것입니다.

이 구조체 데이터를 json으로 변환하려면 Marshal()을 사용하세요. 다음과 같습니다.

post := &Post{1, "Hello World", "userA"}
b, err := json.Marshal(post)
if err != nil {
    fmt.Println(nil)
}
로그인 후 복사

Marshal()은 []바이트 유형을 반환합니다. 이제 변수 b는 출력할 수 있는 []바이트 유형 json 데이터를 저장했습니다.

fmt.Println(string(b))
로그인 후 복사

결과:

{"Id":1,"Content":"Hello World","Author":"userA"}
로그인 후 복사

는 다음과 같이 캡슐화될 수 있습니다. json "beautifying" 시 MarshalIndent()를 사용하여 접두사(접두사 문자열은 일반적으로 비어 있음)와 들여쓰기를 자동으로 추가합니다.

c,err := json.MarshalIndent(post,"","\t")
if err != nil {
    fmt.Println(nil)
}
fmt.Println(string(c))
로그인 후 복사

Result:

{
    "Id": 1,
    "Content": "Hello World",
    "Author": "userA"
}
로그인 후 복사

구조체, 배열, 슬라이스 및 맵 구조 외에도 다음을 수행할 수 있습니다. json으로 파싱되지만, map이 json으로 파싱될 때 키는 json 구문에 필요한 문자열이어야 합니다.

예:

// slice -> json
s := []string{"a", "b", "c"}
d, _ := json.MarshalIndent(s, "", "\t")
fmt.Println(string(d))

// map -> json
m := map[string]string{
    "a":"aa",
    "b":"bb",
    "c":"cc",
}
e,_ := json.MarshalIndent(m,"","\t")
fmt.Println(string(e))
로그인 후 복사

반환 결과:

[
    "a",
    "b",
    "c"
]
{
    "a": "aa",
    "b": "bb",
    "c": "cc"
}
로그인 후 복사

구조체 태그를 사용하여 json 생성을 지원하세요

구조체로 변환할 수 있는 필드는 첫 글자가 대문자인 모든 필드이지만, 원하는 경우 json에서는 소문자를 사용하세요. 키의 경우 구조체의 태그를 사용하여 리플렉션을 지원할 수 있습니다.

예를 들어 Post 구조는 첫 글자가 소문자인 createAt 필드를 추가합니다.

type Post struct {
    Id      int      `json:"ID"`
    Content string   `json:"content"`
    Author  string   `json:"author"`
    Label   []string `json:"label"`
}


postp := &Post{
    2,
    "Hello World",
    "userB",
    []string{"linux", "shell"},
    }

p, _ := json.MarshalIndent(postp, "", "\t")
fmt.Println(string(p))
로그인 후 복사
결과:
{
    "ID": 2,
    "content": "Hello World",
    "author": "userB",
    "label": [
        "linux",
        "shell"
    ]
}
로그인 후 복사
구조 태그를 사용할 때 몇 가지 참고 사항:

1 태그에서 식별된 이름은 json 데이터의 키 값이라고 합니다.

2. to `json:"- "`는 필드 이름의 첫 글자가 대문자인 경우에도 이 필드가 json 데이터로 변환되지 않음을 의미합니다.

json 키의 이름을 "-" 문자로 지정하려면, `json:"-,"`를 특별히 처리할 수 있습니다. 즉, 이전 쉼표

3을 추가할 수 있습니다. 태그에 생략 옵션이 포함된 경우 이 필드의 값이 0, 즉 false, 0, " ", nil 등의 경우 이 필드는 json

4로 변환되지 않습니다. 필드 유형이 bool, string, int class, float class이고 태그에 string 옵션이 있는 경우 이 필드의 값은 다음과 같습니다. json 문자열로 변환

예:

type Post struct {
    Id      int      `json:"ID,string"`
    Content string   `json:"content"`
    Author  string   `json:"author"`
    Label   []string `json:"label,omitempty"`
}
로그인 후 복사

json 데이터를 구조체로 구문 분석합니다(구조는 알려져 있음)

json 데이터는 구조체 또는 빈 인터페이스 인터페이스로 구문 분석할 수 있습니다{}(슬라이싱할 수도 있음, 지도 등). 위에서 json을 구성할 때 태그 규칙을 이해하고 나면 json을 이해하고 파싱하는 것은 매우 간단합니다.

예를 들어 다음은 json 데이터 조각입니다.

{
    "id": 1,
    "content": "hello world",
    "author": {
        "id": 2,
        "name": "userA"
    },
    "published": true,
    "label": [],
    "nextPost": null,
    "comments": [{
            "id": 3,
            "content": "good post1",
            "author": "userB"
        },
        {
            "id": 4,
            "content": "good post2",
            "author": "userC"
        }
    ]
}
로그인 후 복사
이 json 데이터 조각을 분석합니다. 1 최상위 중괄호는 Go의 구조체에 매핑되는 익명 개체를 나타냅니다. struct name은 Post

2. 최상위 중괄호 안의 필드는 모두 Post 구조의 필드이므로 첫 번째 문자를 대문자로 설정해야 합니다. 3. Author는 하위 객체입니다. Mapping to Go는 Post에서 이 필드의 이름이 Author입니다.

4. 레이블은 배열입니다. Go에 매핑되면 슬라이스 또는 배열이 될 수 있으며, json 배열이 비어 있으므로 Go에서는 슬라이스/배열 유형이 가변적일 수 있습니다. 인터페이스{}의 경우 레이블은 string

5여야 하며 nextPost는 Go의 구조체에 매핑되는 하위 개체입니다. 그러나 json의 이 개체는 null이므로 이를 나타냅니다. 객체가 존재하지 않으면 Go에 매핑된 구조체 유형을 결정하는 것이 불가능합니다. 하지만 여기 예에서는 다음 기사가 없으므로 해당 유형도 Post 유형이어야 합니다

6. Comment는 하위 개체이며 Go에 매핑되고 슬라이스/배열입니다. 슬라이스/배열 유형은 구조체입니다

분석 후 그에 따라 구조체와 구조체의 태그를 쉽게 구성할 수 있습니다. 위의 분석을 바탕으로 구성된 데이터 구조는 다음과 같습니다.

type Post struct {
    ID        int64         `json:"id"`       
    Content   string        `json:"content"`  
    Author    Author        `json:"author"`   
    Published bool          `json:"published"`
    Label     []string      `json:"label"`    
    NextPost  *Post         `json:"nextPost"` 
    Comments  []*Comment    `json:"comments"` 
}

type Author struct {
    ID   int64  `json:"id"`  
    Name string `json:"name"`
}

type Comment struct {
    ID      int64  `json:"id"`     
    Content string `json:"content"`
    Author  string `json:"author"` 
}
로그인 후 복사

앞서 json 데이터 구성을 소개할 때 설명한 것처럼 포인터가 추적되므로 여기서 추론한 구조체에서 포인터 유형을 사용해도 문제가 없습니다.

그러므로 이 json 데이터가 .json 파일에 저장되어 있다고 가정하고 위의 json 데이터를 Post 유형 객체로 구문 분석합니다. 코드는 다음과 같습니다:

func main() {
    // 打开json文件
    fh, err := os.Open("a.json")
    if err != nil {
        fmt.Println(err)
        return
    }
    defer fh.Close()
    // 读取json文件,保存到jsonData中
    jsonData, err := ioutil.ReadAll(fh)
    if err != nil {
        fmt.Println(err)
        return
    }
    
    var post Post
    // 解析json数据到post中
    err = json.Unmarshal(jsonData, &post)
    if err != nil {
        fmt.Println(err)
        return
    }
    fmt.Println(post)
}
로그인 후 복사

출력 결과:

{1 hello world {2 userA} true [] <nil> [0xc042072300 0xc0420723c0]}
로그인 후 복사

也许你已经感受到了,从json数据反推算struct到底有多复杂,虽然逻辑不难,但如果数据复杂一点,这是件非常恶心的事情。所以,使用别人写好的工具来自动转换吧。本文后面有推荐json到数据结构的自动转换工具。

解析json到interface(结构未知)

上面是已知json数据结构的解析方式,如果json结构是未知的或者结构可能会发生改变的情况,则解析到struct是不合理的。这时可以解析到空接口interface{}或map[string]interface{}类型上,这两种类型的结果是完全一致的。

解析到interface{}上时,Go类型和JSON类型的对应关系如下

  JSON类型             Go类型                
---------------------------------------------
JSON objects    <-->  map[string]interface{} 
JSON arrays     <-->  []interface{}          
JSON booleans   <-->  bool                   
JSON numbers    <-->  float64                
JSON strings    <-->  string                 
JSON null       <-->  nil
로그인 후 복사

例如:

func main() {
    // 读取json文件
    fh, err := os.Open("a.json")
    if err != nil {
        fmt.Println(err)
        return
    }
    defer fh.Close()
    jsonData, err := ioutil.ReadAll(fh)
    if err != nil {
        fmt.Println(err)
        return
    }
    
    // 定义空接口接收解析后的json数据
    var unknown interface{}
    // 或:map[string]interface{} 结果是完全一样的
    err = json.Unmarshal(jsonData, &unknown)
    if err != nil {
        fmt.Println(err)
        return
    }
    fmt.Println(unknown)
}
로그인 후 복사

输出结果:

map[nextPost:<nil> comments:[map[id:3 content:good post1
author:userB] map[id:4 content:good post2 author:userC]]
id:1 content:hello world author:map[id:2 name:userA] published:true label:[]]
로그인 후 복사

上面将输出map结构。这是显然的,因为类型对应关系中已经说明了,json object解析到Go interface的时候,对应的是map结构。如果将上面输出的结构进行一下格式化,得到的将是类似下面的结构:

map[
    nextPost:<nil>
    comments:[
        map[
            id:3
            content:good post1
            author:userB
        ]
        map[
            id:4
            content:good post2
            author:userC
        ]
    ]
    id:1
    content:hello world
    author:map[
        id:2
        name:userA
    ]
    published:true
    label:[]
]
로그인 후 복사

现在,可以从这个map中去判断类型、取得对应的值。但是如何判断类型?可以使用类型断言:

func main() {
    // 读取json数据
    fh, err := os.Open("a.json")
    if err != nil {
        fmt.Println(err)
        return
    }
    defer fh.Close()
    jsonData, err := ioutil.ReadAll(fh)
    if err != nil {
        fmt.Println(err)
        return
    }
    
    // 解析json数据到interface{}
    var unknown interface{}
    err = json.Unmarshal(jsonData, &unknown)
    if err != nil {
        fmt.Println(err)
        return
    }

    // 进行断言,并switch匹配
    m := unknown.(map[string]interface{})
    for k, v := range m {
        switch vv := v.(type) {
        case string:
            fmt.Println(k, "type: string\nvalue: ", vv)
            fmt.Println("------------------")
        case float64:
            fmt.Println(k, "type: float64\nvalue: ", vv)
            fmt.Println("------------------")
        case bool:
            fmt.Println(k, "type: bool\nvalue: ", vv)
            fmt.Println("------------------")
        case map[string]interface{}:
            fmt.Println(k, "type: map[string]interface{}\nvalue: ", vv)
            for i, j := range vv {
                fmt.Println(i,": ",j)
            }
            fmt.Println("------------------")
        case []interface{}:
            fmt.Println(k, "type: []interface{}\nvalue: ", vv)
            for key, value := range vv {
                fmt.Println(key, ": ", value)
            }
            fmt.Println("------------------")
        default:
            fmt.Println(k, "type: nil\nvalue: ", vv)
            fmt.Println("------------------")
        }
    }
}
로그인 후 복사

结果如下:

comments type: []interface{}
value:  [map[id:3 content:good post1 author:userB] map[author:userC id:4 content:good post2]]
0 :  map[id:3 content:good post1 author:userB]
1 :  map[id:4 content:good post2 author:userC]
------------------
id type: float64
value:  1
------------------
content type: string
value:  hello world
------------------
author type: map[string]interface{}
value:  map[id:2 name:userA]
name :  userA
id :  2
------------------
published type: bool
value:  true
------------------
label type: []interface{}
value:  []
------------------
nextPost type: nil
value:  <nil>
------------------
로그인 후 복사

可见,从interface中解析非常复杂,而且可能因为嵌套结构而导致无法正确迭代遍历。这时候,可以使用第三方包simplejson,见后文。

解析、创建json流

除了可以直接解析、创建json数据,还可以处理流式数据。

1、type Decoder解码json到Go数据结构

2、ype Encoder编码Go数据结构到json

例如:

const jsonStream = `
    {"Name": "Ed", "Text": "Knock knock."}
    {"Name": "Sam", "Text": "Who&#39;s there?"}
    {"Name": "Ed", "Text": "Go fmt."}
    {"Name": "Sam", "Text": "Go fmt who?"}
    {"Name": "Ed", "Text": "Go fmt yourself!"}
`
type Message struct {
    Name, Text string
}
dec := json.NewDecoder(strings.NewReader(jsonStream))
for {
    var m Message
    if err := dec.Decode(&m); err == io.EOF {
        break
    } else if err != nil {
        log.Fatal(err)
    }
    fmt.Printf("%s: %s\n", m.Name, m.Text)
}
로그인 후 복사

输出:

Ed: Knock knock.
Sam: Who&#39;s there?
Ed: Go fmt.
Sam: Go fmt who?
Ed: Go fmt yourself!
로그인 후 복사

再例如,从标准输入读json数据,解码后删除名为Name的元素,最后重新编码后输出到标准输出。

func main() {
    dec := json.NewDecoder(os.Stdin)
    enc := json.NewEncoder(os.Stdout)
    for {
        var v map[string]interface{}
        if err := dec.Decode(&v); err != nil {
            log.Println(err)
            return
        }
        for k := range v {
            if k != "Name" {
                delete(v, k)
            }
        }
        if err := enc.Encode(&v); err != nil {
            log.Println(err)
        }
    }
}
로그인 후 복사

更多go语言知识请关注PHP中文网go语言教程栏目。

위 내용은 Go 언어로 json 데이터를 처리하는 방법의 상세 내용입니다. 자세한 내용은 PHP 중국어 웹사이트의 기타 관련 기사를 참조하세요!

본 웹사이트의 성명
본 글의 내용은 네티즌들의 자발적인 기여로 작성되었으며, 저작권은 원저작자에게 있습니다. 본 사이트는 이에 상응하는 법적 책임을 지지 않습니다. 표절이나 침해가 의심되는 콘텐츠를 발견한 경우 admin@php.cn으로 문의하세요.

핫 AI 도구

Undresser.AI Undress

Undresser.AI Undress

사실적인 누드 사진을 만들기 위한 AI 기반 앱

AI Clothes Remover

AI Clothes Remover

사진에서 옷을 제거하는 온라인 AI 도구입니다.

Undress AI Tool

Undress AI Tool

무료로 이미지를 벗다

Clothoff.io

Clothoff.io

AI 옷 제거제

AI Hentai Generator

AI Hentai Generator

AI Hentai를 무료로 생성하십시오.

인기 기사

R.E.P.O. 에너지 결정과 그들이하는 일 (노란색 크리스탈)
1 몇 달 전 By 尊渡假赌尊渡假赌尊渡假赌
R.E.P.O. 최고의 그래픽 설정
1 몇 달 전 By 尊渡假赌尊渡假赌尊渡假赌
Will R.E.P.O. 크로스 플레이가 있습니까?
1 몇 달 전 By 尊渡假赌尊渡假赌尊渡假赌

뜨거운 도구

메모장++7.3.1

메모장++7.3.1

사용하기 쉬운 무료 코드 편집기

SublimeText3 중국어 버전

SublimeText3 중국어 버전

중국어 버전, 사용하기 매우 쉽습니다.

스튜디오 13.0.1 보내기

스튜디오 13.0.1 보내기

강력한 PHP 통합 개발 환경

드림위버 CS6

드림위버 CS6

시각적 웹 개발 도구

SublimeText3 Mac 버전

SublimeText3 Mac 버전

신 수준의 코드 편집 소프트웨어(SublimeText3)

GO에서 플로팅 포인트 번호 작업에 어떤 라이브러리가 사용됩니까? GO에서 플로팅 포인트 번호 작업에 어떤 라이브러리가 사용됩니까? Apr 02, 2025 pm 02:06 PM

Go Language의 부동 소수점 번호 작동에 사용되는 라이브러리는 정확도를 보장하는 방법을 소개합니다.

Go 's Crawler Colly의 큐 스레드의 문제는 무엇입니까? Go 's Crawler Colly의 큐 스레드의 문제는 무엇입니까? Apr 02, 2025 pm 02:09 PM

Go Crawler Colly의 대기열 스레딩 문제는 Colly Crawler 라이브러리를 GO 언어로 사용하는 문제를 탐구합니다. � ...

이동 중에 왜 println 및 string () 함수로 문자열이 다른 효과를 갖는 이유는 무엇입니까? 이동 중에 왜 println 및 string () 함수로 문자열이 다른 효과를 갖는 이유는 무엇입니까? Apr 02, 2025 pm 02:03 PM

Go Language의 문자열 인쇄의 차이 : println 및 String () 함수 사용 효과의 차이가 진행 중입니다 ...

골란드의 사용자 정의 구조 레이블이 표시되지 않으면 어떻게해야합니까? 골란드의 사용자 정의 구조 레이블이 표시되지 않으면 어떻게해야합니까? Apr 02, 2025 pm 05:09 PM

골란드의 사용자 정의 구조 레이블이 표시되지 않으면 어떻게해야합니까? Go Language 개발을 위해 Goland를 사용할 때 많은 개발자가 사용자 정의 구조 태그를 만날 것입니다 ...

Redis Stream을 사용하여 GO Language에서 메시지 대기열을 구현할 때 User_ID 유형 변환 문제를 해결하는 방법은 무엇입니까? Redis Stream을 사용하여 GO Language에서 메시지 대기열을 구현할 때 User_ID 유형 변환 문제를 해결하는 방법은 무엇입니까? Apr 02, 2025 pm 04:54 PM

Go Language에서 메시지 대기열을 구현하기 위해 Redisstream을 사용하는 문제는 Go Language와 Redis를 사용하는 것입니다 ...

GO의 어떤 라이브러리가 대기업에서 개발하거나 잘 알려진 오픈 소스 프로젝트에서 제공합니까? GO의 어떤 라이브러리가 대기업에서 개발하거나 잘 알려진 오픈 소스 프로젝트에서 제공합니까? Apr 02, 2025 pm 04:12 PM

GO의 어떤 라이브러리가 대기업이나 잘 알려진 오픈 소스 프로젝트에서 개발 했습니까? GO에 프로그래밍 할 때 개발자는 종종 몇 가지 일반적인 요구를 만납니다.

GO 언어에서 'var'와 'type` 키워드 정의 구조의 차이점은 무엇입니까? GO 언어에서 'var'와 'type` 키워드 정의 구조의 차이점은 무엇입니까? Apr 02, 2025 pm 12:57 PM

GO 언어에서 구조를 정의하는 두 가지 방법 : VAR과 유형 키워드의 차이. 구조를 정의 할 때 Go Language는 종종 두 가지 다른 글쓰기 방법을 본다 : 첫째 ...

sql.open을 사용할 때 DSN이 비어있을 때 오류가 발생하지 않습니까? sql.open을 사용할 때 DSN이 비어있을 때 오류가 발생하지 않습니까? Apr 02, 2025 pm 12:54 PM

sql.open을 사용할 때 DSN에 오류가 발생하지 않는 이유는 무엇입니까? Go Language, SQL.open ...

See all articles