API Call Expecting Array, получение объекта - PullRequest
0 голосов
/ 17 апреля 2019

У меня есть вызов API, который ищет массив и объекты в основном в одном месте. Я пытаюсь вызвать API и получаю следующую ошибку. io.reactivex.exceptions.OnErrorNotImplementedException: Expected BEGIN_ARRAY but was BEGIN_OBJECT at line 1 column 2 path $

Как я могу вызвать как корневые элементы, так и вложенные элементы в классе атрибутов?

У меня есть некоторые входы в приложение, и URL-адрес правильный, но ответ - это проблема

Это API -

{
    "id": "65",
    "name": "Switch - Kitchen",
    "label": "Switch - Kitchen",
    "attributes": [
        {
            "name": "switch",
            "currentValue": "off",
            "dataType": "ENUM",
            "values": [
                "on",
                "off"
            ]
        }
    ],
    "capabilities": [
        "Switch",
        {
            "attributes": [
                {
                    "name": "switch",
                    "dataType": null
                }
            ]
        },
        "Configuration",
        "Refresh",
        "Actuator"
    ],
    "commands": [
        "configure",
        "flash",
        "off",
        "on",
        "refresh",
        "refresh"
    ]
}

Используя инструмент отображения json, это то, что он определяет для макета

public class Attribute {

@SerializedName("name")
@Expose
private String name;
@SerializedName("currentValue")
@Expose
private String currentValue;
@SerializedName("dataType")
@Expose
private String dataType;
@SerializedName("values")

отдельный класс, который он определил

public class Example {

@SerializedName("id")
@Expose
private String id;
@SerializedName("name")
@Expose
private String name;
@SerializedName("label")
@Expose
private String label;
@SerializedName("attributes")
@Expose
private List<Attribute> attributes = null;
@SerializedName("capabilities")
@Expose
private List<String> capabilities = null;
@SerializedName("commands")
@Expose
private List

Информация об устройстве Class

data class DeviceDetails(
    val attributes: List<Attribute>,
    val capabilities: List<String>,
    val commands: List<String>,
    val id: String,
    val label: String,
    val name: String
)

Класс атрибутов

data class Attribute(
    val currentValue: String,
    val dataType: String,
    val name: String,
    val values: List<String>
)

Адаптер интерфейса

interface INetworkAPIDetails {
    @GET("devices/65")
    fun getAllDetails(@Query("access_token") access_token: String): Observable<List<Attribute>>

Adapater

class PostItemDetailsAdapter(private val postdetailsList: List<Attribute>, private val context: Context) :
    RecyclerView.Adapter<PostItemDetailsAdapter.ViewHolder>() {

    override fun onCreateViewHolder(parent: ViewGroup, viewType: Int): ViewHolder {
        return ViewHolder(
            LayoutInflater.from(context).inflate(R.layout.post_item_details,
                parent, false))
    }

    override fun getItemCount(): Int {
        return postdetailsList.size
    }

    override fun onBindViewHolder(holder: ViewHolder, position: Int) {
        holder.itemView.txtPostName.text = this.postdetailsList[position].name
        holder.itemView.txtPostCurrent.text = this.postdetailsList[position].currentValue
        holder.itemView.txtPostDataType.text = this.postdetailsList[position].dataType

    }
    class ViewHolder(view: View) : RecyclerView.ViewHolder(view)
}

деятельности2

class Activity2 : AppCompatActivity() {

    override fun onCreate(savedInstanceState: Bundle?) {
        super.onCreate(savedInstanceState)
        setContentView(R.layout.activity2)

        //Debugging URL//
        val interceptor : HttpLoggingInterceptor = HttpLoggingInterceptor().apply {
            this.level = HttpLoggingInterceptor.Level.BODY
        }

        val client : OkHttpClient = OkHttpClient.Builder().apply {
            this.addInterceptor(interceptor)
        }.build()
        //Debugging URL//

        Log.d("TAG", "Activity 2")

        val retrofit = Retrofit.Builder().addConverterFactory(GsonConverterFactory.create(GsonBuilder().create()))
            .addCallAdapterFactory(RxJava2CallAdapterFactory.create())
            .baseUrl("http://xxx.xxx.xxx.xxx/apps/api/109/")
            .client(client)
            .build()
        //Base of API Call//


        //Build List of Devices//
        val postsApi = retrofit.create(INetworkAPIDetails::class.java)
        var response = postsApi.getAllDetails(access_token = "xxxx")

        //Displaying List//
        response.observeOn(AndroidSchedulers.mainThread()).subscribeOn(IoScheduler()).subscribe {
            rv__list_posts_details.adapter = PostItemDetailsAdapter(it, this)
        }
    }
}

1 Ответ

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

Формируйте JSON, которым вы поделились, ваш INetworkAPIDetails должен быть:

interface INetworkAPIDetails {
    @GET("devices/65")
    fun getAllDetails(@Query("access_token") access_token: String): Observable<DeviceDetails>
}

Структура выглядит как DeviceDetails, плюс List означало, что ожидал массив объектов, перед исправлением массив объектов типа Attribute.

Кроме того, у вас есть сложный случай в массиве capabilities: ваша структура данных определяется как List<String> ... но второй элемент "Switch" и "Configuration" не является String, это полный Object.

Для этого случая см. Как анализировать массив json с несколькими объектами по gson? или Использование GSON для анализа массива с несколькими типами

...