質問をすることでしか得られない、回答やアドバイスがある。

15分調べてもわからないことは、質問しよう!

ただいまの
回答率

89.10%

【GO】構造体を扱った、JSON生成で困っています

解決済

回答 2

投稿

  • 評価
  • クリップ 0
  • VIEW 613

Gunjirk

score 17

実現したいこと

以下のJSONを出力する

//理想のJSON
{
  "command": "ENT",
  "table": [
    {
      "table_name": "basic",
      "clientid": 0,
      "server": "basic"
    },
    {
      "table_name": "contents",
      "clientid": 0,
      "time_control": "0"
    }
   ]
}
//現在のJSON
{
    "command": "ENT",
    "table": [
        {
            "Table1": {
                "table_name": "basic",
                "clientid": 0,
                "server": "basic"
            },
            "Table2": {
                "table_name": "contents",
                "clientid": 0,
                "time_control": 0
            }
        }
    ]
}

"Table1": {
"Table2": {
の部分が不要なのですが、どうしても消せません。。。

プログラムは以下です。
異なる構造体をまとめて、リテラルスライスで出力しようとしています。

package main

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

//Command ENT
type Command struct {
    Command string  `json:"command"`
    Tables  []Table `json:"table"`
}

//Table Data
type Table1 struct {
    TableName string `json:"table_name"`
    Clientid  int    `json:"clientid"`
    Server    string `json:"server"`
}

type Table2 struct {
    TableName   string `json:"table_name"`
    Clientid    int    `json:"clientid"`
    TimeControl int    `json:"time_control"`
}

type Table struct {
    Table1 Table1
    Table2 Table2
}

func main() {
    BasicConfig := Table1{TableName: "basic", Clientid: 0, Server: "basic"}
    ContentsFunc := Table2{TableName: "contents", Clientid: 0, TimeControl: 0}
    jsonStr := Command{
        Command: "ENT",
        Tables: []Table{
            {
                BasicConfig,
                ContentsFunc,
            },
        },
    }

    jsonBytes, err := json.Marshal(jsonStr)
    if err != nil {
        fmt.Println("JSON Marshal error:", err)
        return
    }
    out := new(bytes.Buffer)
    // プリフィックスなし、スペース4つでインデント
    json.Indent(out, jsonBytes, "", "    ")
    fmt.Println(out.String())
}

よろしくおねがいします。

  • 気になる質問をクリップする

    クリップした質問は、後からいつでもマイページで確認できます。

    またクリップした質問に回答があった際、通知やメールを受け取ることができます。

    クリップを取り消します

  • 良い質問の評価を上げる

    以下のような質問は評価を上げましょう

    • 質問内容が明確
    • 自分も答えを知りたい
    • 質問者以外のユーザにも役立つ

    評価が高い質問は、TOPページの「注目」タブのフィードに表示されやすくなります。

    質問の評価を上げたことを取り消します

  • 評価を下げられる数の上限に達しました

    評価を下げることができません

    • 1日5回まで評価を下げられます
    • 1日に1ユーザに対して2回まで評価を下げられます

    質問の評価を下げる

    teratailでは下記のような質問を「具体的に困っていることがない質問」、「サイトポリシーに違反する質問」と定義し、推奨していません。

    • プログラミングに関係のない質問
    • やってほしいことだけを記載した丸投げの質問
    • 問題・課題が含まれていない質問
    • 意図的に内容が抹消された質問
    • 過去に投稿した質問と同じ内容の質問
    • 広告と受け取られるような投稿

    評価が下がると、TOPページの「アクティブ」「注目」タブのフィードに表示されにくくなります。

    質問の評価を下げたことを取り消します

    この機能は開放されていません

    評価を下げる条件を満たしてません

    評価を下げる理由を選択してください

    詳細な説明はこちら

    上記に当てはまらず、質問内容が明確になっていない質問には「情報の追加・修正依頼」機能からコメントをしてください。

    質問の評価を下げる機能の利用条件

    この機能を利用するためには、以下の事項を行う必要があります。

回答 2

checkベストアンサー

+1

JSONに対応するオブジェクトでよく勘違いされるパターンのひとつなんですが、
似て異なるものを別の型として表現しようとすることです。
実際はフィールドがオミットされているだけで同じ型を使っているという解釈で済む場合が多々あります。

そもそもTableというアレイフィールドに
異なる型のオブジェクトを入れ込むような設計はかなり稀です。

JSONの各要素の型を設計をするときはできるだけ型を決めて
その「あるPath」に対する「値の型」は一定であるという設計がお勧めですし、
そうなっていないJSON設計は良くない設計だと思います。

今回の事例も以下のように一つのTable型で済みます。

https://play.golang.org/p/qj9DPdI7qmV

package main

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

type Command struct {
    Command string  `json:"command"`
    Table   []Table `json:"table"`
}

type Table struct {
    TableName   string `json:"table_name"`
    Clientid    int    `json:"clientid"`
    Server      string `json:"server,omitempty"`
    TimeControl string `json:"time_control,omitempty"`
}

func main() {
    BasicConfig := Table{TableName: "basic", Clientid: 0, Server: "basic", TimeControl:""}
    ContentsFunc := Table{TableName: "contents", Clientid: 0, Server:"", TimeControl: "0"}
    jsonStr := Command{
        Command: "ENT",
        Table: []Table{
            BasicConfig,
            ContentsFunc,
        },
    }

    jsonBytes, err := json.Marshal(jsonStr)
    if err != nil {
        fmt.Println("JSON Marshal error:", err)
        return
    }
    out := new(bytes.Buffer)
    // プリフィックスなし、スペース4つでインデント
    json.Indent(out, jsonBytes, "", "    ")
    fmt.Println(out.String())
}

投稿

編集

  • 回答の評価を上げる

    以下のような回答は評価を上げましょう

    • 正しい回答
    • わかりやすい回答
    • ためになる回答

    評価が高い回答ほどページの上位に表示されます。

  • 回答の評価を下げる

    下記のような回答は推奨されていません。

    • 間違っている回答
    • 質問の回答になっていない投稿
    • スパムや攻撃的な表現を用いた投稿

    評価を下げる際はその理由を明確に伝え、適切な回答に修正してもらいましょう。

  • 2020/06/29 16:53

    ちなみに2つのオミットフィールドの「両方に有効な値がある」「両方がエンプティである」のはNGであるという場合のチェックの責任は「JSONデコード結果を受け取る側」または「JSONエンコードに値を渡す側」の責任です。

    キャンセル

  • 2020/06/29 17:00

    nobonoboさん、丁寧なご回答ありがとうございます。
    無事に解決いたしました。

    nobonoboさんのおっしゃる勘違いをしておりました。
    もう少しGOを勉強していきます。
    JSON設計に関しても見直していきます。

    今後ともよろしくお願いします。

    キャンセル

+1

nobonobo さんの回答のとおり、両方のsrtuctのフィールドをもつstructを用いて処理するのが良いと思います。

あえて、汎用的に何でも受け取るようにしたいのであれば、以下のように []interface{} で受けてしまうのも最終手段の方法としてはあるかな、と思います。

//Command ENT
type Command struct {
    Command string        `json:"command"`
    Tables  []interface{} `json:"table"`
}

https://play.golang.org/p/ZVOxE5qHOnX

  • 出力結果
{
    "command": "ENT",
    "table": [
        {
            "table_name": "basic",
            "clientid": 0,
            "server": "basic"
        },
        {
            "table_name": "contents",
            "clientid": 0,
            "time_control": 0
        }
    ]
}

投稿

  • 回答の評価を上げる

    以下のような回答は評価を上げましょう

    • 正しい回答
    • わかりやすい回答
    • ためになる回答

    評価が高い回答ほどページの上位に表示されます。

  • 回答の評価を下げる

    下記のような回答は推奨されていません。

    • 間違っている回答
    • 質問の回答になっていない投稿
    • スパムや攻撃的な表現を用いた投稿

    評価を下げる際はその理由を明確に伝え、適切な回答に修正してもらいましょう。

  • 2020/06/29 17:03

    d_tutuzさん、引き続きご回答いただきありがとうございます。

    interfaceも使えるのですね!
    試してみたいと思います!

    キャンセル

15分調べてもわからないことは、teratailで質問しよう!

  • ただいまの回答率 89.10%
  • 質問をまとめることで、思考を整理して素早く解決
  • テンプレート機能で、簡単に質問をまとめられる