วิธีการใช้แจ็คสันในการกำจัดวัตถุต่างๆออก


779

เอกสารผูกพันข้อมูลแจ็คสันระบุว่าแจ็คสันสนับสนุน deserialising "อาร์เรย์ชนิดที่สนับสนุนทั้งหมด" แต่ฉันไม่สามารถคิดออกไวยากรณ์ที่แน่นอนสำหรับการนี้

สำหรับวัตถุเดียวฉันจะทำสิ่งนี้:

//json input
{
    "id" : "junk",
    "stuff" : "things"
}

//Java
MyClass instance = objectMapper.readValue(json, MyClass.class);

ตอนนี้สำหรับอาร์เรย์ฉันต้องการทำสิ่งนี้:

//json input
[{
    "id" : "junk",
    "stuff" : "things"
},
{
    "id" : "spam",
    "stuff" : "eggs"
}]

//Java
List<MyClass> entries = ?

ใครรู้ว่ามีคำสั่งหายไปมายากล? ถ้าไม่เช่นนั้นทางออกคืออะไร?


2
ฉันชอบห้องสมุด GSON ของ Google เพื่อจัดการกับ JSON เป็นมูลค่าการตรวจสอบว่าคุณยังไม่ได้ลอง ... ทำให้การทำงานกับมันง่ายและใช้งานง่าย
Jesse Webb

11
FWIW การแก้ปัญหาที่เป็นไปได้สำหรับปัญหาเฉพาะนี้กับ Gson นั้นเกือบจะเหมือนกับสิ่งที่เป็นไปได้ด้วย Data Binding API ของ Jackson
โปรแกรมเมอร์ Bruce

17
Gweebz - บางทีคุณอยากจะอธิบายว่าทำไมคุณถึงรู้สึกว่า GSON เป็นตัวเลือกที่ดีกว่า (เทียบกับ Jackson)
StaxMan

คำตอบ:


1656

สร้าง mapper แรก:

import com.fasterxml.jackson.databind.ObjectMapper;// in play 2.3
ObjectMapper mapper = new ObjectMapper();

เป็นอาร์เรย์:

MyClass[] myObjects = mapper.readValue(json, MyClass[].class);

เป็นรายการ:

List<MyClass> myObjects = mapper.readValue(jsonInput, new TypeReference<List<MyClass>>(){});

อีกวิธีในการระบุประเภทรายการ:

List<MyClass> myObjects = mapper.readValue(jsonInput, mapper.getTypeFactory().constructCollectionType(List.class, MyClass.class));

43
หมายเหตุเพิ่มเติมหนึ่งรายการหากในขณะที่วิเคราะห์คำคุณได้รับข้อผิดพลาดเช่นJsonMappingException: No suitable constructor found for typeนั้นหมายความว่าคุณต้องเพิ่มตัวสร้างเริ่มต้นให้กับคลาสของคุณโดยเพิ่มตัวสร้างแบบไม่มีอาร์กิวเมนต์ส่วนตัวแก้ไขให้ฉัน
SyntaxRules

12
@SyntaxRules การเพิ่ม Constructor ที่ชัดเจนเป็นสิ่งจำเป็นหากคุณมี Constructor ที่ชัดเจน - หากไม่มีคอมไพเลอร์จะสร้าง Constructor "Empty" สาธารณะโดยอัตโนมัติ จุดดี. ปัญหาที่พบบ่อยอีกประการหนึ่งก็คือต้องมีการเรียนภายในstatic- มิฉะนั้นพวกเขาไม่เคยมีคอนสตรัคศูนย์หาเรื่อง
StaxMan

244
Btw List<MyClass> myObjects = Arrays.asList(mapper.readValue(json, MyClass[].class))ทำงานได้เร็วกว่า TypeRefence ถึง 10 เท่า

5
ฉันกำลังมองหารุ่นทั่วไป
Stephane

1
@EugeneTskhovrebov ทางของคุณสะอาดที่สุดสำหรับการทำอินไลน์ คนอื่น ๆ ต้องการการหล่อหรือการประกาศ ฉันขอแนะนำให้เพิ่มเป็นคำตอบของตัวเองเพื่อช่วยเน้นและให้ตัวแทนคุณ
Erik B

190

จากEugene Tskhovrebov

List<MyClass> myObjects = Arrays.asList(mapper.readValue(json, MyClass[].class))

วิธีนี้ดูเหมือนจะดีที่สุดสำหรับฉัน


สำหรับผู้ที่ทำงานกับ Agent ใน Java, Lotus Domino นี่คือวิธีที่จะไป ฉันลองวิธีแก้ไขปัญหาอื่น ๆ แต่ได้รับเสมอResourceNotFoundException
John

SyntaxRules นอกจากนี้ในความคิดเห็นสำหรับคำตอบข้างต้นอาจจำเป็นสำหรับการแก้ปัญหานี้เพราะเรามันเป็นของฉัน ฉันแค่อยากจะเพิ่มมันเพื่อไม่ให้สูญหาย
Rob

2
หรือArrays.asList(Json.fromJson(json.get("fieldName"), MyClass[].class))
Al-Mothafar

3
หรือList<MyClass> myObjects = Arrays.asList(mapper.treeToValue(jsonNode.get("fieldName"), MyClass[].class))
Collin Krawll

@CollinKrawll objectmapper.treetovalue ทำอะไรได้บ้าง
Eswar

35

สำหรับการใช้งานทั่วไป:

public static <T> List<T> parseJsonArray(String json,
                                         Class<T> classOnWhichArrayIsDefined) 
                                         throws IOException, ClassNotFoundException {
   ObjectMapper mapper = new ObjectMapper();
   Class<T[]> arrayClass = (Class<T[]>) Class.forName("[L" + classOnWhichArrayIsDefined.getName() + ";");
   T[] objects = mapper.readValue(json, arrayClass);
   return Arrays.asList(objects);
}

4
โครงสร้างที่ดีของ Class <T []> ไม่เคยเห็นสิ่งนี้ คุณหาข้อมูลเกี่ยวกับสิ่งนี้ได้จากที่ไหน
Roeland Van Heddegem

11

สร้างอินสแตนซ์ของ ObjectReader ซึ่งปลอดภัยต่อเธรด

ObjectMapper objectMapper = new ObjectMapper();
ObjectReader objectReader = objectMapper.reader().forType(new TypeReference<List<MyClass>>(){});

จากนั้นใช้:

List<MyClass> result = objectReader.readValue(inputStream);

นี่คือสิ่งที่ฉันกำลังมองหาขอบคุณ
ยืนอยู่คนเดียว

เราได้รับ - com.fasterxml.jackson.databind.JsonMappingException: ไม่สามารถทำการ deserialize ตัวอย่างของ java.util.ArrayList จากโทเค็น START_OBJECT ที่ [ที่มา: java.io.FileInputStream@33fec21; บรรทัด: 1, คอลัมน์: 1]
Dinesh Kumar

7
try {
    ObjectMapper mapper = new ObjectMapper();
    JsonFactory f = new JsonFactory();
    List<User> lstUser = null;
    JsonParser jp = f.createJsonParser(new File("C:\\maven\\user.json"));
    TypeReference<List<User>> tRef = new TypeReference<List<User>>() {};
    lstUser = mapper.readValue(jp, tRef);
    for (User user : lstUser) {
        System.out.println(user.toString());
    }

} catch (JsonGenerationException e) {
    e.printStackTrace();
} catch (JsonMappingException e) {
    e.printStackTrace();
} catch (IOException e) {
    e.printStackTrace();
}

3

นี่คือยูทิลิตี้ที่สามารถแปลง json2object หรือ Object2json ไม่ว่า pojo ของคุณ (เอนทิตี T)

import java.io.IOException;
import java.io.StringWriter;
import java.util.List;

import com.fasterxml.jackson.core.JsonGenerationException;
import com.fasterxml.jackson.core.JsonParseException;
import com.fasterxml.jackson.core.type.TypeReference;
import com.fasterxml.jackson.databind.DeserializationFeature;
import com.fasterxml.jackson.databind.JsonMappingException;
import com.fasterxml.jackson.databind.ObjectMapper;
import com.fasterxml.jackson.databind.SerializationFeature;

/**
 * 
 * @author TIAGO.MEDICI
 * 
 */
public class JsonUtils {

    public static boolean isJSONValid(String jsonInString) {
        try {
            final ObjectMapper mapper = new ObjectMapper();
            mapper.readTree(jsonInString);
            return true;
        } catch (IOException e) {
            return false;
        }
    }

    public static String serializeAsJsonString(Object object) throws JsonGenerationException, JsonMappingException, IOException {
        ObjectMapper objMapper = new ObjectMapper();
        objMapper.enable(SerializationFeature.INDENT_OUTPUT);
        objMapper.disable(SerializationFeature.FAIL_ON_EMPTY_BEANS);
        StringWriter sw = new StringWriter();
        objMapper.writeValue(sw, object);
        return sw.toString();
    }

    public static String serializeAsJsonString(Object object, boolean indent) throws JsonGenerationException, JsonMappingException, IOException {
        ObjectMapper objMapper = new ObjectMapper();
        if (indent == true) {
            objMapper.enable(SerializationFeature.INDENT_OUTPUT);
            objMapper.disable(SerializationFeature.FAIL_ON_EMPTY_BEANS);
        }

        StringWriter stringWriter = new StringWriter();
        objMapper.writeValue(stringWriter, object);
        return stringWriter.toString();
    }

    public static <T> T jsonStringToObject(String content, Class<T> clazz) throws JsonParseException, JsonMappingException, IOException {
        T obj = null;
        ObjectMapper objMapper = new ObjectMapper();
        obj = objMapper.readValue(content, clazz);
        return obj;
    }

    @SuppressWarnings("rawtypes")
    public static <T> T jsonStringToObjectArray(String content) throws JsonParseException, JsonMappingException, IOException {
        T obj = null;
        ObjectMapper mapper = new ObjectMapper();
        obj = mapper.readValue(content, new TypeReference<List>() {
        });
        return obj;
    }

    public static <T> T jsonStringToObjectArray(String content, Class<T> clazz) throws JsonParseException, JsonMappingException, IOException {
        T obj = null;
        ObjectMapper mapper = new ObjectMapper();
        mapper = new ObjectMapper().configure(DeserializationFeature.ACCEPT_SINGLE_VALUE_AS_ARRAY, true);
        obj = mapper.readValue(content, mapper.getTypeFactory().constructCollectionType(List.class, clazz));
        return obj;
    }

0

คุณสามารถสร้างคลาสที่ขยายArrayList:

public static class MyList extends ArrayList<Myclass> {}

แล้วใช้มันเช่น:

List<MyClass> list = objectMapper.readValue(json, MyList.class);

0

ฉันไม่สามารถใช้คำตอบนี้ได้เนื่องจาก linter ของฉันจะไม่อนุญาตการปลดเปลื้องที่ไม่ถูกตรวจสอบ

นี่คือทางเลือกที่คุณสามารถใช้ได้ ฉันรู้สึกว่ามันเป็นทางออกที่สะอาดกว่า

public <T> List<T> parseJsonArray(String json, Class<T> clazz) throws JsonProcessingException {
  var tree = objectMapper.readTree(json);
  var list = new ArrayList<T>();
  for (JsonNode jsonNode : tree) {
    list.add(objectMapper.treeToValue(jsonNode, clazz));
  }
  return list;
}
โดยการใช้ไซต์ของเรา หมายความว่าคุณได้อ่านและทำความเข้าใจนโยบายคุกกี้และนโยบายความเป็นส่วนตัวของเราแล้ว
Licensed under cc by-sa 3.0 with attribution required.