JsonMappingException: нет токена START_ARRAY

112

Учитывая следующий файл .json:

[
    {
        "name" : "New York",
        "number" : "732921",
        "center" : [
                "latitude" : 38.895111, 
                "longitude" : -77.036667
            ]
    },
    {
        "name" : "San Francisco",
        "number" : "298732",
        "center" : [
                "latitude" : 37.783333, 
                "longitude" : -122.416667
            ]
    }
]

Я подготовил два класса для представления содержащихся данных:

public class Location {
    public String name;
    public int number;
    public GeoPoint center;
}

...

public class GeoPoint {
    public double latitude;
    public double longitude;
}

Чтобы проанализировать содержимое файла .json, я использую Jackson 2.2.x и подготовил следующий метод:

public static List<Location> getLocations(InputStream inputStream) {
    ObjectMapper objectMapper = new ObjectMapper();
    try {
        TypeFactory typeFactory = objectMapper.getTypeFactory();
        CollectionType collectionType = typeFactory.constructCollectionType(
                                            List.class, Location.class);
        return objectMapper.readValue(inputStream, collectionType);
    } catch (IOException e) {
        e.printStackTrace();
    }
    return null;
}

Пока я не указываю centerсвойство, все содержимое можно анализировать. Однако, когда я пытаюсь разобрать географические координаты, я получаю следующее сообщение об ошибке:

com.fasterxml.jackson.databind.JsonMappingException: невозможно десериализовать экземпляр
com.example.GeoPoint из маркера START_ARRAY в [Источник: android.content.res.AssetManager$AssetInputStream@416a5850; строка: 5, столбец: 25]
(через цепочку ссылок: com.example.Location ["center"])

JJD
источник
зачем вы используете Джексона, если вы можете просто разобрать его с помощью парсера json
Шашанк Агарвал
4
Ваша строка JSON имеет неправильный формат, тип center- массив недопустимых объектов. Попробуйте заменить [и ]на {и }в строке JSON вокруг, longitudeи тогда latitudeони будут объектами.
Katona
1
@Katona Спасибо. Не могли бы вы превратить ваш комментарий в ответ, чтобы я мог закрыть вопрос ?!
JJD

Ответы:

76

Ваша строка JSON имеет неправильный centerформат: это массив недопустимых объектов. Замените [и ]на {и }в строке JSON вокруг, longitudeи latitudeони будут объектами:

[
    {
        "name" : "New York",
        "number" : "732921",
        "center" : {
                "latitude" : 38.895111, 
                "longitude" : -77.036667
            }
    },
    {
        "name" : "San Francisco",
        "number" : "298732",
        "center" : {
                "latitude" : 37.783333, 
                "longitude" : -122.416667
            }
    }
]
Катона
источник
2
Та же проблема, что и у меня, но безумная, так как в jackson 2.6.3v была совершенно другая сумасшедшая ошибка, а с 2.9.8v все работало. Черт побери, так эти ошибки сериализации так раздражают.
brebDev
115

JsonMappingException: out of START_ARRAY tokenИсключение генерируется сопоставителем объектов Джексона, поскольку он ожидает, когда Object {}он нашел Array [{}]в ответ.

Это может быть решено путем замены Objectс Object[]в аргументе для geForObject("url",Object[].class). Ссылки:

  1. Ссылка 1
  2. Ссылка 2
  3. Ссылка 3
Абхиджит
источник
9

Я отсортировал эту проблему как проверку json с JSONLint.com, а затем исправил ее. И это код того же.

String jsonStr = "[{\r\n" + "\"name\":\"New York\",\r\n" + "\"number\": \"732921\",\r\n"+ "\"center\": {\r\n" + "\"latitude\": 38.895111,\r\n"  + " \"longitude\": -77.036667\r\n" + "}\r\n" + "},\r\n" + " {\r\n"+ "\"name\": \"San Francisco\",\r\n" +\"number\":\"298732\",\r\n"+ "\"center\": {\r\n" + "    \"latitude\": 37.783333,\r\n"+ "\"longitude\": -122.416667\r\n" + "}\r\n" + "}\r\n" + "]";

ObjectMapper mapper = new ObjectMapper();
MyPojo[] jsonObj = mapper.readValue(jsonStr, MyPojo[].class);

for (MyPojo itr : jsonObj) {
    System.out.println("Val of name is: " + itr.getName());
    System.out.println("Val of number is: " + itr.getNumber());
    System.out.println("Val of latitude is: " + 
        itr.getCenter().getLatitude());
    System.out.println("Val of longitude is: " + 
        itr.getCenter().getLongitude() + "\n");
}

Примечание: MyPojo[].classэто класс, имеющий геттер и установщик свойств json.

Результат:

Val of name is: New York
Val of number is: 732921
Val of latitude is: 38.895111
Val of longitude is: -77.036667
Val of name is: San Francisco
Val of number is: 298732
Val of latitude is: 37.783333
Val of longitude is: -122.416667
Атул Шарма
источник
1
Это устранило проблему, с которой я сталкивался несколько дней, другие ответы не имели смысла. Для всех, у кого такая же ошибка. Стоит отметить, что если ваш JSON содержит несколько объектов, вам необходимо возвращать несколько объектов. Я возвращал MyObject, а должен был вернуть MyObject []
Аария
8

Как уже говорилось, JsonMappingException: out of START_ARRAY tokenобъект сопоставления объектов Jackson генерирует исключение, поскольку он ожидает, в Object {}то время как он нашел Array [{}]в ответ.

Более простым решением может быть замена метода getLocationsна:

public static List<Location> getLocations(InputStream inputStream) {
    ObjectMapper objectMapper = new ObjectMapper();
    try {
        TypeReference<List<Location>> typeReference = new TypeReference<>() {};
        return objectMapper.readValue(inputStream, typeReference);
    } catch (IOException e) {
        e.printStackTrace();
    }
    return null;
}

С другой стороны, если у вас нет pojo like Location, вы можете использовать:

TypeReference<List<Map<String, Object>>> typeReference = new TypeReference<>() {};
return objectMapper.readValue(inputStream, typeReference);
Freedev
источник
1
Спасибо, что поделились этой альтернативной реализацией. Можете ли вы сказать, есть ли преимущества или недостатки в использовании CollectionTypevs. TypeReference?
JJD 05
1
TypeReferenceсинтаксис намного короче другого. Не уверен, если в некоторых случаях может быть контр-указание, связанное со стиранием типа ... но в моем мире TypeReferenceэто просто проще использовать и понять.
freedev