Ожидаемый STRING, но был BEGIN_OBJECT - PullRequest
1 голос
/ 17 апреля 2019

Я прочитал много ответов об этой ошибке, и я не могу найти решение для моей, может быть, объект класса не тот же

Моя ошибка:

com.google.gson.JsonSyntaxException: java.lang.IllegalStateException: ожидаемая STRING, но была BEGIN_OBJECT

Итак, я просто пытаюсь получить объект из JSON следующим образом:

JsonObject mS1SystemStatusJsonObject = parameterEntries.getValue().getAsJsonObject();

MS1SystemStatusResponse mS1SystemStatusResponse = gson.fromJson(mS1SystemStatusJsonObject, MS1SystemStatusResponse.class);

мой объект JsonObject:

{
 "WARNING_CONDITION_LIST":
  {
   "warningConditionList": {"warningCondition":"BLACK_POSTAL_INK_WF_COND"}
   },
 "DISABLING_CONDITION_LIST": {"disablingConditionList":""},
 "CurrentErrorList":"",
 "LockState":"lockspend",
"SystemReadyState":true
}

Мои отличия POJO (я не ставлю get и задал для краткости код):

public class MS1SystemStatusResponse{

@SerializedName("LockState")
String lockState;

@SerializedName("SystemReadyState")
Boolean systemReadyState;

@SerializedName("WARNING_CONDITION_LIST")
WARNING_CONDITION_LIST warningConditionListMy;

@SerializedName("DISABLING_CONDITION_LIST")
DISABLING_CONDITION_LIST disablingConditionListMy;

@JsonAdapter(EmptyStringAsNullTypeAdapter.class)    
@SerializedName("CurrentErrorList")
CurrentErrorList currentErrorList;


public class WARNING_CONDITION_LIST {

@SerializedName(value = "warningConditionList")
@JsonAdapter(EmptyStringAsNullTypeAdapter.class)
WarningConditionList warningConditionList;


public class DISABLING_CONDITION_LIST {

@SerializedName(value = "disablingConditionList")
@JsonAdapter(EmptyStringAsNullTypeAdapter.class)
DisablingConditionList disablingConditionList;

public class CurrentErrorList {

@SerializedName(value = "Error")
private Error error;


public class WARNING_CONDITION_LIST {

@SerializedName(value = "warningConditionList")
@JsonAdapter(EmptyStringAsNullTypeAdapter.class)
WarningConditionList warningConditionList;


public class WarningConditionList {

@SerializedName(value = "warningCondition")
String warningCondition;

Вот полная трассировка стека и другой класс, который у меня естьзабыл добавить раньше:

Stacktrace:

java.lang.IllegalStateException: Expected STRING but was BEGIN_OBJECT
    com.google.gson.internal.bind.JsonTreeReader.nextString(JsonTreeReader.java:154)
    com.pb.marqueo.connect.plus.api.common.model.response.EmptyStringAsNullTypeAdapter.read(EmptyStringAsNullTypeAdapter.java:24)
    com.pb.marqueo.connect.plus.api.common.model.response.EmptyStringAsNullTypeAdapter.read(EmptyStringAsNullTypeAdapter.java:10)
    com.google.gson.internal.bind.ReflectiveTypeAdapterFactory$1.read(ReflectiveTypeAdapterFactory.java:117)
    com.google.gson.internal.bind.ReflectiveTypeAdapterFactory$Adapter.read(ReflectiveTypeAdapterFactory.java:217)
    com.google.gson.internal.bind.ReflectiveTypeAdapterFactory$1.read(ReflectiveTypeAdapterFactory.java:117)
    com.google.gson.internal.bind.ReflectiveTypeAdapterFactory$Adapter.read(ReflectiveTypeAdapterFactory.java:217)
    com.google.gson.Gson.fromJson(Gson.java:861)
    com.google.gson.Gson.fromJson(Gson.java:926)
    com.google.gson.Gson.fromJson(Gson.java:899)
    com.pb.marqueo.connect.plus.api.core.service.impl.ConnectPlusServicesImpl.getMS1SystemStatus(ConnectPlusServicesImpl.java:410)
    com.pb.marqueo.connect.plus.api.core.facade.impl.ConnectPlusBusinessServicesImpl.startProduction(ConnectPlusBusinessServicesImpl.java:1236)
    com.pb.marqueo.connect.plus.rest.services.impl.ConnectPlusControllerImpl.startProd(ConnectPlusControllerImpl.java:280)
    sun.reflect.NativeMethodAccessorImpl.invoke0(Native Method)
    sun.reflect.NativeMethodAccessorImpl.invoke(NativeMethodAccessorImpl.java:62)
    sun.reflect.DelegatingMethodAccessorImpl.invoke(DelegatingMethodAccessorImpl.java:43)
    java.lang.reflect.Method.invoke(Method.java:498)
    org.springframework.web.method.support.InvocableHandlerMethod.invoke(InvocableHandlerMethod.java:215)
    org.springframework.web.method.support.InvocableHandlerMethod.invokeForRequest(InvocableHandlerMethod.java:132)
    org.springframework.web.servlet.mvc.method.annotation.ServletInvocableHandlerMethod.invokeAndHandle(ServletInvocableHandlerMethod.java:104)
    org.springframework.web.servlet.mvc.method.annotation.RequestMappingHandlerAdapter.invokeHandleMethod(RequestMappingHandlerAdapter.java:749)
    org.springframework.web.servlet.mvc.method.annotation.RequestMappingHandlerAdapter.handleInternal(RequestMappingHandlerAdapter.java:689)
    org.springframework.web.servlet.mvc.method.AbstractHandlerMethodAdapter.handle(AbstractHandlerMethodAdapter.java:83)
    org.springframework.web.servlet.DispatcherServlet.doDispatch(DispatcherServlet.java:938)
    org.springframework.web.servlet.DispatcherServlet.doService(DispatcherServlet.java:870)
    org.springframework.web.servlet.FrameworkServlet.processRequest(FrameworkServlet.java:961)
    org.springframework.web.servlet.FrameworkServlet.doPost(FrameworkServlet.java:863)
    javax.servlet.http.HttpServlet.service(HttpServlet.java:644)
    org.springframework.web.servlet.FrameworkServlet.service(FrameworkServlet.java:837)
    javax.servlet.http.HttpServlet.service(HttpServlet.java:725)
    org.apache.tomcat.websocket.server.WsFilter.doFilter(WsFilter.java:52)
    com.pb.marqueo.common.filter.OriginFilter.doFilter(OriginFilter.java:70)

Адаптер класса:

public class EmptyStringAsNullTypeAdapter extends TypeAdapter<String> {

    @Override
    public void write(JsonWriter out, String value) throws IOException {
        out.value(value);
    }

    @Override
    public String read(JsonReader in) throws IOException {
        if (in.peek() == JsonToken.NULL) {
            in.nextNull();
            return null;
        }

        String result = in.nextString();
        if ("".equals(result)) {
            return null;
        }
        return result;
    }
}

Оборачиваюсь с 1 недели: (

1 Ответ

1 голос
/ 18 апреля 2019

Хорошо, поэтому из разговора в комментариях я понял, что вы хотите иметь возможность десериализовать оба этих случая:

{
  "WARNING_CONDITION_LIST": {
      "warningConditionList": {"warningCondition":"BLACK_POSTAL_INK_WF_COND"}
  },
  ...
}

и

{
  "WARNING_CONDITION_LIST": ""
  ...
}

В первом сценарии вы хотите объект, а во втором сценарии вы хотите null.

Первое, что я заметил, это то, что у вас аннотация @JsonAdapter не в том месте. На самом деле он должен быть в поле WARNING_CONDITION_LIST. Примерно так:

public class MS1SystemStatusResponse {
  // ...
  @SerializedName("WARNING_CONDITION_LIST")
  @JsonAdapter(EmptyStringAsNullTypeAdapter.class)
  WARNING_CONDITION_LIST warningConditionListMy;
  // ...
}

Однако одно это не решит вашу проблему. Проблема в пределах EmptyStringAsNullTypeAdapter. Он ожидает, что поле будет просто строкой, и не обрабатывает случай, когда он может быть объектом. Для этого мы можем написать следующий адаптер:

public class EmptyStringAsEmptyWarningConditionlTypeAdapter extends TypeAdapter<WARNING_CONDITION_LIST> {

  @Override
  public void write(JsonWriter out, WARNING_CONDITION_LIST value) throws IOException {
    out.beginObject();
    out.name("warningConditionList");
    out.value(value.warningConditionList.warningCondition);
    out.endObject();
  }

  @Override
  public WARNING_CONDITION_LIST read(JsonReader in) throws IOException {
    WARNING_CONDITION_LIST result = new WARNING_CONDITION_LIST();
    result.warningConditionList =  new WarningConditionList();
    result.warningConditionList.warningCondition = "";

    if (in.peek() == JsonToken.BEGIN_OBJECT) {
        in.beginObject();
        in.nextName();
        in.beginObject();
        in.nextName();
        result.warningConditionList.warningCondition = in.nextString();
        in.endObject();
        in.endObject();
    } else {
        in.nextString();
        return null;
    }
    return result;
  }
}

По сути, он проверит, есть ли у вас строка или объект, и вернет null или объект соответственно.

Это работает только для этих двух случаев, которые я показал вам с помощью json, и должно быть достаточно, чтобы помочь вам разобраться с остальными.

Возможно, вы столкнетесь с аналогичной проблемой с CurrentErrorList.

...