Jackson을 사용하여 객체 배열을 직렬화 해제하는 방법


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
JSON을 처리하기 위해 Google의 GSON 라이브러리를 선호합니다. 아직 시도하지 않았다면 확인해 볼 가치가 있습니다 ... 작업을 매우 쉽고 직관적으로 만듭니다.
Jesse Webb

11
FWIW Gson의이 특정 문제에 대한 가능한 솔루션은 Jackson의 데이터 바인딩 API에서 가능한 것과 거의 동일합니다.
프로그래머 Bruce

17
Gweebz-왜 GSON이 더 나은 선택이라고 생각하는지 설명하고 싶습니까 (잭슨과 비교)?
StaxMan

답변:


1656

먼저 매퍼를 만듭니다.

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가 필요합니다. 그렇지 않은 경우 컴파일러는 자동으로 "빈"생성자를 생성합니다. 좋은 지적. 또 다른 일반적인 문제는 내부 클래스가 필요하다는 것입니다 static. 그렇지 않으면 인수가 0 인 생성자가 없습니다.
StaxMan

244
Btw List<MyClass> myObjects = Arrays.asList(mapper.readValue(json, MyClass[].class))는 TypeRefence보다 최대 10 배 빠르게 작동합니다.

5
일반 유형 버전을 찾고 있습니다.
Stephane

1
@EugeneTskhovrebov 당신의 방법은 인라인을 위해 가장 깨끗합니다. 다른 것들은 캐스팅이나 선언이 필요합니다. 나는 그것을 강조하고 도움을 줄 수 있도록 자체 답변으로 추가하는 것이 좋습니다.
Erik B

190

에서 유진 Tskhovrebov

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

이 솔루션은 나에게 가장 좋은 것 같습니다


Java, Lotus Domino에서 에이전트 작업을 수행하는 사람들에게이 방법이 사용됩니다. 다른 솔루션을 시도했지만 항상 다음을 얻었습니다.ResourceNotFoundException
John

이 솔루션에는 위의 답변에 대한 주석에 규칙 추가가 필요할 수 있습니다. 나는 그것을 잃어 버리지 않도록 추가하고 싶었습니다.
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 : [Source : java.io.FileInputStream@33fec21;의 START_OBJECT 토큰에서 java.util.ArrayList의 인스턴스를 직렬화 해제 할 수 없습니다. 줄 : 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

여기에 당신의 pojo (엔티티 T)에 관계없이 json2object 또는 Object2json을 변환하는 유틸리티가 있습니다.

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

내 린터가 확인되지 않은 캐스트를 허용하지 않기 때문에이 답변 을 사용할 수 없습니다 .

사용할 수있는 대안은 다음과 같습니다. 실제로 그것이 더 깨끗한 해결책이라고 생각합니다.

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.