แปลงสตริง JSON ให้เป็นเอาต์พุตพริ้นท์ 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 Print JSON Output (โดยใช้แจ็คสัน) เช่นด้านล่าง:

{
    "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
        }
    ]
}

ใครช่วยให้ฉันตัวอย่างตามตัวอย่างของฉันข้างต้น วิธีการบรรลุสถานการณ์นี้? ฉันรู้ว่ามีตัวอย่างมากมาย แต่ฉันไม่สามารถเข้าใจสิ่งเหล่านั้นได้อย่างถูกต้อง ความช่วยเหลือใด ๆ จะได้รับการชื่นชมด้วยตัวอย่างง่ายๆ

Updated:

ด้านล่างเป็นรหัสที่ฉันใช้:

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

แต่วิธีนี้ใช้ไม่ได้กับวิธีที่ฉันต้องการเอาต์พุตตามที่ระบุไว้ข้างต้น

นี่คือ POJO ที่ฉันใช้สำหรับ JSON ด้านบน:

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;
}

มีใครบอกฉันได้ไหมว่าฉันได้รับ POJO ที่ถูกต้องสำหรับ JSON หรือไม่

Updated:

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));

จากนั้นแสดงให้เห็นในหน้าผลลัพธ์ของ 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 Tree) ได้เช่นกัน


ขอบคุณ StaxMan ฉันคิดว่ามันใช้งานได้ เมื่อฉันพิมพ์ออกมาเยื้องฉันได้รับในแบบที่ฉันต้องการ แต่เมื่อฉันใช้การเยื้องเพื่อเพิ่มลงในโมเดลเพื่อให้ฉันสามารถแสดงพวกเขาในหน้าผลลัพธ์ มันยังคงได้รับการพิมพ์ในสองสามบรรทัด ฉันได้อัปเดตคำถามบางทีคุณอาจจะได้รับความคิดเพิ่มเติมว่าเกิดอะไรขึ้นตอนนี้
คลังแสง

ปัญหาเกิดขึ้นกับ Spring แล้ว - ฉันคิดว่ามันน่าจะเป็น POJO เป็นคุณลักษณะไม่ใช่สตริงที่จัดรูปแบบไว้ล่วงหน้า ดังนั้นแทนที่จะพยายามจัดรูปแบบด้วยตัวคุณเองคุณต้องบอกให้ Spring ทำเช่นนี้ เมื่อใช้ Jackson ควรกำหนดค่าให้ใช้การเยื้อง แม้ว่าจะซื่อสัตย์ฉันไม่แน่ใจว่าทำไมคุณถึงต้องเยื้องมันเพื่อตอบสนอง
StaxMan

30
สวัสดี defaultPrettyPrintingWriter () เลิกใช้แล้ว ตั้งแต่ 1.9 ใช้ writerWithDefaultPrettyPrinter () แทน อ้างอิง: jackson.codehaus.org/1.9.0/javadoc/org/codehaus/jackson/map/…
Lin

6
สำหรับ Jackson 2 ใช้ SerializationFeature.INDENT_OUTPUT ตามที่ระบุโดย Marcelo C. ด้านล่าง
Mike R

ความคิดวิธีการเผยแพร่จำนวนเงินที่เขียนโดยใช้แจ็คสัน?
Reyansh Mishra

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 อื่น
Innokenty

@Innokenty จากนั้นข้ามบรรทัดแรก
muttonUp

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) ในtoString()วิธีการ

หมายเหตุ:มันทำงานได้ดีใน Android โดยไม่ต้องมีห้องสมุดใด ๆ แต่ในจาวาคุณต้องใช้ไลบรารีorg.json


3
น่าสังเกตว่านี่เป็นการใช้ไลบรารีJSON ใน Java (org.json)
Steve Chambers

ใน Android มันสามารถใช้ direclty ได้โดยไม่ต้องใช้ libraray
Aman Gupta - ShOoTeR

String json = new GsonBuilder().setPrettyPrinting().create().toJson(map); String indentedJson = (new JSONObject(json)).toString(4);ด้วยเหตุผลบางอย่างที่สองคือการสูญเสียคำสั่งของคีย์
Michail Michailidis

วิธีการในปัจจุบันน่าเสียดายที่ไม่ได้จัดการรายการของวัตถุ json ฉันหมายถึง [{"id": "1"}, {"id": "2"}]
Geniy

4

นี้ดูเหมือนว่ามันอาจจะเป็นคำตอบสำหรับคำถามของคุณ มันบอกว่ากำลังใช้สปริง แต่ฉันคิดว่ามันน่าจะช่วยคุณได้ ขอให้ฉันใส่โค้ดไว้ที่นี่เพื่อความสะดวกมากขึ้น:

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;}
}

ขอบคุณ Daniel สำหรับความช่วยเหลือ ส่วนที่ยากที่สุดที่ฉันมีคือการทำตัวแบบ JSON ของฉันเป็นคลาสได้อย่างไร ถ้าฉันทำงานส่วนนั้นฉันก็สามารถโค้ดส่วนที่เหลือได้อย่างง่ายดาย
คลังแสง

คุณช่วยดูคลาส POJO ของฉันที่ฉันเขียนจาก JSON ได้ไหม มันดูถูกหรือไม่?
คลังแสง

3

คุณสามารถทำได้โดยใช้วิธีการร้อง:

1. การใช้ Jackson จาก Apache

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

นำเข้าชั้นร้อง:

import com.fasterxml.jackson.databind.ObjectMapper;

มันขึ้นอยู่กับการไล่ระดับคือ:

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. การใช้ Gson จาก Google

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() สามารถทำได้ในหนึ่งบรรทัด:

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

1
เหตุผลที่ฉันชอบคำตอบนี้ก็คือมันไม่ได้ทำการแปลงออบเจ็กต์ใด ๆ เลยนอกจากการจับคู่โดยตรงกับประเภท JSON ตราบใดที่สตริงอินพุตเป็น JSON ที่ถูกต้องเรารู้ว่าสตริงเอาต์พุตจะเทียบเท่า JSON เชิงความหมาย
M. Justin
โดยการใช้ไซต์ของเรา หมายความว่าคุณได้อ่านและทำความเข้าใจนโยบายคุกกี้และนโยบายความเป็นส่วนตัวของเราแล้ว
Licensed under cc by-sa 3.0 with attribution required.