JSON

JSON #

// Go підтримує кодування та розкодування формату JSON
// на рівні, як базових типів даних, так і власних
// структур даних, так би кажучи "з коробки".

package main

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

// Ми скористаємось цими структурами для демонстрації
// кодування та розкодування нетипових для Go структур даних.
type response1 struct {
	Page   int
	Fruits []string
}

// Тільки експортовані поля будуть закодовані/розкодовані у JSON.
// Поля повинні починатись з великої літери, щоб бути "експортованими".
type response2 struct {
	Page   int      `json:"page"`
	Fruits []string `json:"fruits"`
}

func main() {

	// Для початку ми розглянемо кодування базових
	// типів, доступних у Go, в формат JSON.
	bolB, _ := json.Marshal(true)
	fmt.Println(string(bolB))

	intB, _ := json.Marshal(1)
	fmt.Println(string(intB))

	fltB, _ := json.Marshal(2.34)
	fmt.Println(string(fltB))

	strB, _ := json.Marshal("gopher")
	fmt.Println(string(strB))

	// В цьому прикладі ми працюємо з коллекціями
	// - зрізами та мапами, що будуть кодуватись в JSON
	// об'єкти передбачуваним (стандартизованим) чином.
	slcD := []string{"apple", "peach", "pear"}
	slcB, _ := json.Marshal(slcD)
	fmt.Println(string(slcB))

	mapD := map[string]int{"apple": 5, "lettuce": 7}
	mapB, _ := json.Marshal(mapD)
	fmt.Println(string(mapB))

	// Пакет JSON автоматично кодує ваші типи даних.
	// Це дозволяє автоматично включати експортовані
	// поля до вашого виводу, або навпаки - виключати
	// не-експортовані з цього виводу. По-замовчуванню,
	// ці елементи будуть носити тіж назви, що і поля
	// структури яка кодувалась.
	res1D := &response1{
		Page:   1,
		Fruits: []string{"apple", "peach", "pear"}}
	res1B, _ := json.Marshal(res1D)
	fmt.Println(string(res1B))

	// Ви можете скористатись тегами або метаданим для
	// декларації полів у вашій структурі даних, гнучко налаштовуючи
	// логіку відображення або споживання даних за ключами.
	res2D := &response2{
		Page:   1,
		Fruits: []string{"apple", "peach", "pear"}}
	res2B, _ := json.Marshal(res2D)
	fmt.Println(string(res2B))

	// Тепер, розглянемо як декодувати дані з формату JSON
	// у значення Go, на прикладі узагальненої структури.
	byt := []byte(`{"num":6.13,"strs":["a","b"]}`)

	// Ми починаємо з того, що декларуємо змінну, якою пакет
	// JSON скористається для збереження розкодованих даних.
	// Змінна `dat` з типом `map[string]interface{}` буде зберігати
	// дані у вигляді мапи з рядковим ключем та значенням довільного типу.
	var dat map[string]interface{}

	// Розкодуємо наш JSON та проведемо перевірку на можливі помилки.
	if err := json.Unmarshal(byt, &dat); err != nil {
		panic(err)
	}
	fmt.Println(dat)

	// Для використання даних розкодованої мапи, нам необхідно буде
	// конвертувати (англійською `cast`) дані (тому що значення
	// нашої мапи - початково не типізовані - `interface{}`), щоб
	// продовжити роботу вже з чітко визначеним типом,
	// що підтримується Go. Для прикладу ми конвертуємо значення
	// мапи за ключем `num` в тип `float64`.
	num := dat["num"].(float64)
	fmt.Println(num)

	// Доступ до вкладених даних - потребує серії конвертацій.
	strs := dat["strs"].([]interface{})
	str1 := strs[0].(string)
	fmt.Println(str1)

	// Слід використовувати ваші власні структури даних для
	// розкодування даних, це гарантія додаткової типо-безпеки,
	// що прибирає необхідність перевіряти типи під час використання.
	str := `{"page": 1, "fruits": ["apple", "peach"]}`
	res := response2{}
	json.Unmarshal([]byte(str), &res)
	fmt.Println(res)
	fmt.Println(res.Fruits[0])

	// У, вже розглянутих, прикладах ми використовуємо зрізи байтів
	// та рядки як посередники між даними та JSON репрезентацією
	// для виводу до стандартного потоку даних. Подивіться, як
	// можна скористатись перенаправленням потоку за допомогою
	// власної реалізацію інтерфейсу `Writer`, наприклад `os.Writer`,
	// або навіть запис у HTTP Response.
	enc := json.NewEncoder(os.Stdout)
	d := map[string]int{"apple": 5, "lettuce": 7}
	enc.Encode(d)
}
$ go run json.go
true
1
2.34
"gopher"
["apple","peach","pear"]
{"apple":5,"lettuce":7}
{"Page":1,"Fruits":["apple","peach","pear"]}
{"page":1,"fruits":["apple","peach","pear"]}
map[num:6.13 strs:[a b]]
6.13
a
{1 [apple peach]}
apple
{"apple":5,"lettuce":7}


# Ми ознайомились з базовою роботою Go з форматом
# JSON, і радимо прочитати статтю
# [JSON та Go](http://blog.golang.org/2011/01/json-and-go.html)
# та документацію [пакету JSON](http://golang.org/pkg/encoding/json/)
# для отримання більш розгорнутої інформації про сам
# пакет в Go.