Jacksonを使用してJSON文字列をPretty Print JSON出力に変換する


165

これは私が持っているJSON文字列です:

{"attributes":[{"nm":"ACCOUNT","lv":[{"v":{"Id":null,"State":null},"vt":"java.util.Map","cn":1}],"vt":"java.util.Map","status":"SUCCESS","lmd":13585},{"nm":"PROFILE","lv":[{"v":{"Party":null,"Ads":null},"vt":"java.util.Map","cn":2}],"vt":"java.util.Map","status":"SUCCESS","lmd":41962}]}

以下のように、上記のJSON StringをPretty Jackson JSON出力(Jacksonを使用)に変換する必要があります。

{
    "attributes": [
        {
            "nm": "ACCOUNT",
            "lv": [
                {
                    "v": {
                        "Id": null,
                        "State": null
                    },
                    "vt": "java.util.Map",
                    "cn": 1
                }
            ],
            "vt": "java.util.Map",
            "status": "SUCCESS",
            "lmd": 13585
        },
        {
            "nm": "PROFILE
            "lv": [
                {
                    "v": {
                        "Party": null,
                        "Ads": null
                    },
                    "vt": "java.util.Map",
                    "cn": 2
                }
            ],
            "vt": "java.util.Map",
            "status": "SUCCESS",
            "lmd": 41962
        }
    ]
}

上記の私の例に基づいて誰かが私に例を提供できますか?このシナリオを達成する方法は?多くの例があることは知っていますが、それらを正しく理解することができません。簡単な例を参考にしてください。

更新しました:

以下は私が使用しているコードです:

ObjectMapper mapper = new ObjectMapper();
System.out.println(mapper.defaultPrettyPrintingWriter().writeValueAsString(jsonString));

しかし、これは上記の出力が必要な方法では機能しません。

以下は、上記のJSONに使用しているPOJOです。

public class UrlInfo implements Serializable {

    private List<Attributes> attribute;

}

class Attributes {

    private String nm;
    private List<ValueList> lv;
    private String vt;
    private String status;
    private String lmd;

}


class ValueList {
    private String vt;
    private String cn;
    private List<String> v;
}

JSONに適切なPOJOを取得したかどうかを誰かが教えてくれますか?

更新しました:

String result = restTemplate.getForObject(url.toString(), String.class);

ObjectMapper mapper = new ObjectMapper();
Object json = mapper.readValue(result, Object.class);

String indented = mapper.defaultPrettyPrintingWriter().writeValueAsString(json);

System.out.println(indented);//This print statement show correct way I need

model.addAttribute("response", (indented));

下の行は次のようなものを出力します:

System.out.println(indented);


{
  "attributes" : [ {
    "nm" : "ACCOUNT",
    "error" : "null SYS00019CancellationException in CoreImpl fetchAttributes\n java.util.concurrent.CancellationException\n\tat java.util.concurrent.FutureTask$Sync.innerGet(FutureTask.java:231)\n\tat java.util.concurrent.FutureTask.",
    "status" : "ERROR"
  } ]
}

これが私が示す必要がある方法です。しかし、次のようにモデルに追加すると、

model.addAttribute("response", (indented));

そして、それを以下のようなresultform jspページで示します:

    <fieldset>
        <legend>Response:</legend>
            <strong>${response}</strong><br />

    </fieldset>

私はこのようなものを手に入れます:

{ "attributes" : [ { "nm" : "ACCOUNT", "error" : "null    
SYS00019CancellationException in CoreImpl fetchAttributes\n 
java.util.concurrent.CancellationException\n\tat 
java.util.concurrent.FutureTask$Sync.innerGet(FutureTask.java:231)\n\tat 
java.util.concurrent.FutureTask.", "status" : "ERROR" } ] }

私には必要ありません。私はそれが上に印刷される方法が必要でした。なぜそれがこのように起こったのか誰かに教えてもらえますか?

回答:


252

古いJSONをインデントするにはObject、次のようにバインドします。

Object json = mapper.readValue(input, Object.class);

そして、インデントでそれを書きます:

String indented = mapper.writerWithDefaultPrettyPrinter().writeValueAsString(json);

これにより、データをマップする実際のPOJOを定義する必要がなくなります。

または、JsonNode(JSONツリー)も使用できます。


StaxManに感謝します。これは機能していると思います。字下げして印刷すると、必要な方法で邪魔になります。しかし、インデントを使用してモデルに追加すると、それらを結果フォームページに表示できます。それでも、2行3行で印刷されます。質問を更新しました。おそらく、今何が起こっているかについてもう少し理解できるでしょう。
アーセナル2013年

そのときの問題はSpringにあります-事前にフォーマットされた文字列ではなく、属性としてPOJOを期待していると思います。したがって、自分でフォーマットするのではなく、これを行うようにSpringに指示する必要があります。ジャクソンを使用する場合、インデントを使用するように構成できるはずです。正直なところ、なぜそれをインデントして応答する必要があるのか​​よくわかりません。
StaxMan 2013年

30
こんにちは、defaultPrettyPrintingWriter()は非推奨になりました。1.9以降では、代わりにwriterWithDefaultPrettyPrinter()を使用してください。参照してください:jackson.codehaus.org/1.9.0/javadoc/org/codehaus/jackson/map/...
なブラウン林

6
ジャクソン2の場合、以下のMarcelo Cによって指定されているSerializationFeature.INDENT_OUTPUTを使用します
Mike R

ジャクソンを使用して書かれた価値の量を公開する方法はありますか?
Reyansh Mishra 2017

125

最もシンプルで最もコンパクトなソリューション(v2.3.3の場合):

ObjectMapper mapper = new ObjectMapper();
mapper.enable(SerializationFeature.INDENT_OUTPUT);
mapper.writeValueAsString(obj)

21
実際にはさらに短くすることができます。ObjectMapper mapper = new ObjectMapper.enable(SerializationFeature.INDENT_OUTPUT);
Jason Nichols

26

Jackson 1.9以降を使用する新しい方法は次のとおりです。

Object json = OBJECT_MAPPER.readValue(diffResponseJson, Object.class);
String indented = OBJECT_MAPPER.writerWithDefaultPrettyPrinter()
                               .writeValueAsString(json);

出力は正しくフォーマットされます!


1
残念ながら、私の入力がランタイムで作成されたオブジェクトであり、別のjsonではない場合、これは役に立ちません。
イノケンティ2015

@Innokentyその後、最初の行をスキップします。
muttonUp 2015

2
@muttonUpええ、明らかに。私はそれさえしました、なぜ私がそのような愚かなコメントを残したのか分かりません=)
Innokenty

17

Jackson 1.9では、次のコードを使用してきれいに印刷できます。

ObjectMapper objectMapper = new ObjectMapper();
objectMapper.enable(SerializationConfig.Feature.INDENT_OUTPUT);

11

これは、jsonデータを美化する最も簡単な手法だと思います。

String indented = (new JSONObject(Response)).toString(4);

ここで、Responseは文字列です。

4(indentSpaces)in toString()メソッドを渡すだけです。

注:ライブラリがなくてもAndroidで正常に動作します。しかし、Javaではorg.jsonライブラリを使用する必要があります。


3
これは、JSON in Java(org.json)ライブラリを使用していることに注目する価値があります
Steve Chambers

Androidでは、ライブラリを使用せずに直接使用できます。
アマングプタ-ShOoTeR

String json = new GsonBuilder().setPrettyPrinting().create().toJson(map); String indentedJson = (new JSONObject(json)).toString(4);何らかの理由で、2番目のキーがキーの順序を失っています
Michail Michailidis

残念ながら、現在のアプローチはjsonオブジェクトのリストを処理しません。
つまり

4

これはあなたの質問に対する答えかもしれないようです。それはSpringを使用していると書いてありますが、それでもあなたの場合には役立つはずです。ここにコードをインライン化して、より便利にします。

import java.io.FileReader;

import org.codehaus.jackson.map.ObjectMapper;
import org.codehaus.jackson.map.ObjectWriter;

public class Foo
{
  public static void main(String[] args) throws Exception
  {
    ObjectMapper mapper = new ObjectMapper();
    MyClass myObject = mapper.readValue(new FileReader("input.json"), MyClass.class);
    // this is Jackson 1.x API only: 
    ObjectWriter writer = mapper.defaultPrettyPrintingWriter();
    // ***IMPORTANT!!!*** for Jackson 2.x use the line below instead of the one above: 
    // ObjectWriter writer = mapper.writer().withDefaultPrettyPrinter();
    System.out.println(writer.writeValueAsString(myObject));
  }
}

class MyClass
{
  String one;
  String[] two;
  MyOtherClass three;

  public String getOne() {return one;}
  void setOne(String one) {this.one = one;}
  public String[] getTwo() {return two;}
  void setTwo(String[] two) {this.two = two;}
  public MyOtherClass getThree() {return three;}
  void setThree(MyOtherClass three) {this.three = three;}
}

class MyOtherClass
{
  String four;
  String[] five;

  public String getFour() {return four;}
  void setFour(String four) {this.four = four;}
  public String[] getFive() {return five;}
  void setFive(String[] five) {this.five = five;}
}

ダニエルに助けてくれてありがとう。私が持っている最も難しい部分は、JSONをクラスにモデル化する方法です。その部分を機能させると、残りの部分を簡単にコーディングできます。
アーセナル2013年

JSONから作成したPOJOクラスを確認できますか?正しく見えますか?
アーセナル2013年

3

これは、次の方法で実現できます。

1. ApacheからJacksonを使用する

    String formattedData=new ObjectMapper().writerWithDefaultPrettyPrinter()
.writeValueAsString(YOUR_JSON_OBJECT);

怒鳴るクラスをインポート:

import com.fasterxml.jackson.databind.ObjectMapper;

それのgradle依存関係は次のとおりです:

compile 'com.fasterxml.jackson.core:jackson-core:2.7.3'
compile 'com.fasterxml.jackson.core:jackson-annotations:2.7.3'
compile 'com.fasterxml.jackson.core:jackson-databind:2.7.3'

2. GoogleのGsonを使用する

String formattedData=new GsonBuilder().setPrettyPrinting()
    .create().toJson(YOUR_OBJECT);

怒鳴るクラスをインポート:

import com.google.gson.Gson;

グラドルは次のとおりです。

compile 'com.google.code.gson:gson:2.8.2'

ここでは、リポジトリから更新された正しいバージョンをダウンロードできます。


2

ObjectMapper.readTree() これを1行で行うことができます。

mapper.writerWithDefaultPrettyPrinter().writeValueAsString(mapper.readTree(json));

1
私がこの回答を気に入っている理由は、JSON型への直接マッピング以外はオブジェクト変換をまったく行わないためです。入力文字列が有効なJSONである限り、出力文字列は意味的に同等のJSONになることがわかっています。
M.ジャスティン
弊社のサイトを使用することにより、あなたは弊社のクッキーポリシーおよびプライバシーポリシーを読み、理解したものとみなされます。
Licensed under cc by-sa 3.0 with attribution required.