GSON "BEGIN_OBJECT की अपेक्षा" लेकिन BEGIN_ARRAY फेंक रहा था?


295

मैं इस तरह एक JSON स्ट्रिंग पार्स करने की कोशिश कर रहा हूँ

[
   {
      "updated_at":"2012-03-02 21:06:01",
      "fetched_at":"2012-03-02 21:28:37.728840",
      "description":null,
      "language":null,
      "title":"JOHN",
      "url":"http://rus.JOHN.JOHN/rss.php",
      "icon_url":null,
      "logo_url":null,
      "id":"4f4791da203d0c2d76000035",
      "modified":"2012-03-02 23:28:58.840076"
   },
   {
      "updated_at":"2012-03-02 14:07:44",
      "fetched_at":"2012-03-02 21:28:37.033108",
      "description":null,
      "language":null,
      "title":"PETER",
      "url":"http://PETER.PETER.lv/rss.php",
      "icon_url":null,
      "logo_url":null,
      "id":"4f476f61203d0c2d89000253",
      "modified":"2012-03-02 23:28:57.928001"
   }
]

वस्तुओं की सूची में।

List<ChannelSearchEnum> lcs = (List<ChannelSearchEnum>) new Gson().fromJson( jstring , ChannelSearchEnum.class);

यहाँ एक ऑब्जेक्ट क्लास का उपयोग कर रहा हूँ।

import com.google.gson.annotations.SerializedName;

public class ChannelSearchEnum {



@SerializedName("updated_at")
private String updated_at;

@SerializedName("fetched_at")
private String fetched_at;

@SerializedName("description")
private String description;

@SerializedName("language")
private String language;

@SerializedName("title")
private String title;

@SerializedName("url")
private String url;

@SerializedName("icon_url")
private String icon_url;

@SerializedName("logo_url")
private String logo_url;

@SerializedName("id")
private String id;

@SerializedName("modified")
private String modified;

public final String get_Updated_at() {
    return this.updated_at;
}

public final String get_Fetched_at() {
    return this.fetched_at;
}

public final String get_Description() {
    return this.description;
}

public final String get_Language() {
    return this.language;
}

public final String get_Title() {
    return this.title;
}

public final String get_Url() {
    return this.url;
}

public final String get_Icon_url() {
    return this.icon_url;
}

public final String get_Logo_url() {
    return this.logo_url;
}

public final String get_Id() {
    return this.id;
}

public final String get_Modified() {
    return this.modified;
}

        }

लेकिन यह मेरे साथ फेंकता है

com.google.gson.JsonSyntaxException: java.lang.IllegalStateException: Expected BEGIN_OBJECT but was BEGIN_ARRAY at line 1 column 2

किसी भी विचार मुझे इसे कैसे ठीक करना चाहिए?


12
@ सोनी - यह गलत है। यदि आप jsonlint.org पर जाते हैं और उसके JSON को कॉपी / पेस्ट करते हैं, तो आप देखेंगे कि यह मान्य है।
ब्रायन रोच

@ सोनी - नहीं, हटा दिया गया "[" और "]", लेकिन अभी भी वही है। लगता है कि यह अधिक हो सकता है क्योंकि स्ट्रिंग में मेरे पास एक नहीं बल्कि कई ऑब्जेक्ट हैं।
रोजर ट्रैविस

आपका क्या jstringदिखता है जो आपने अपने कोड में दिया था?
इगोरगानापोलस्की

मैं एक सोचता हूं, जब प्रतिक्रिया सरणी में वापस आती है तो सूची में लेने की कोशिश करें, यह मेरी समस्या का समाधान है।
इम्काडब्लू

जवाबों:


331

समस्या यह है कि आप बता रहे हैं कि Gsonआपके पास अपने प्रकार की कोई वस्तु है। तुम नहीं। आपके पास अपने प्रकार की वस्तुओं की एक सरणी है। आप बस कोशिश नहीं कर सकते और उस तरह का परिणाम डाल सकते हैं और इसे जादुई रूप से काम करने की उम्मीद कर सकते हैं;)

इस Gsonसे निपटने के तरीके की व्याख्या के लिए उपयोगकर्ता गाइड :

https://github.com/google/gson/blob/master/UserGuide.md

यह काम करेगा:

ChannelSearchEnum[] enums = gson.fromJson(yourJson, ChannelSearchEnum[].class);

लेकिन यह बेहतर है:

Type collectionType = new TypeToken<Collection<ChannelSearchEnum>>(){}.getType();
Collection<ChannelSearchEnum> enums = gson.fromJson(yourJson, collectionType);

शायद। ऑब्जेक्ट की एक सरणी के रूप में, प्रकार को रनटाइम पर बनाए रखा जाता है इसलिए गॉन जानता है कि उसे क्या देखना है। अच्छा विचार।
njzk2

3
+1 के लिए TypoToken<Collection<Something>>- जब आप संग्रह (उपवर्ग) और / या Iterables रख सकते हैं तो सरणियों का उपयोग न करें।
फिलिप रीचर्ट

क्या आपको लगता है कि चयनित obj / array को पार्स करना सही तरीका है? मदद stackoverflow.com/questions/18140830/…
LOG_TAG

1
क्या होगा अगर हम इसे स्ट्रिंग के साथ बनाना चाहते हैं; उदाहरण के लिए क्या मैं स्ट्रिंग [] t = gson.fromJson (myJson, String [] वर्ग] जैसे कुछ लिख सकता हूं
साहिन यान्लीक

4
लगता है यह उत्तर अधूरा है !!
इंजनविज़न

45

समस्या यह है कि आप एक प्रकार की वस्तु के लिए पूछ रहे हैं, ChannelSearchEnumलेकिन वास्तव में आपके पास एक प्रकार की वस्तु है List<ChannelSearchEnum>

आप इसे प्राप्त कर सकते हैं:

Type collectionType = new TypeToken<List<ChannelSearchEnum>>(){}.getType();
List<ChannelSearchEnum> lcs = (List<ChannelSearchEnum>) new Gson()
               .fromJson( jstring , collectionType);

1
वह किस प्रकार का Typeहै? क्या आयात करें?
smatthewenglish

4
@ सबसे अधिक संभावना S.Matthew_Englishjava.lang.reflect.Type
Guillaume Polet

36

मेरे मामले में JSON स्ट्रिंग:

[{"category":"College Affordability",
  "uid":"150151",
  "body":"Ended more than $60 billion in wasteful subsidies for big banks and used the savings to put the cost of college within reach for more families.",
  "url":"http:\/\/www.whitehouse.gov\/economy\/middle-class\/helping middle-class-families-pay-for-college",
  "url_title":"ending subsidies for student loan lenders",
  "type":"Progress",
  "path":"node\/150385"}]

और मैं रीसायकलव्यू में "श्रेणी" और "url_title" प्रिंट करता हूं

Datum.class

import com.google.gson.annotations.Expose;
import com.google.gson.annotations.SerializedName;

public class Datum {
@SerializedName("category")
@Expose
private String category;
@SerializedName("uid")
@Expose
private String uid;
@SerializedName("url_title")
@Expose
private String urlTitle;

/**
 * @return The category
 */
public String getCategory() {
    return category;
}

/**
 * @param category The category
 */
public void setCategory(String category) {
    this.category = category;
}

/**
 * @return The uid
 */
public String getUid() {
    return uid;
}

/**
 * @param uid The uid
 */
public void setUid(String uid) {
    this.uid = uid;
}

/**
 * @return The urlTitle
 */
public String getUrlTitle() {
    return urlTitle;
}

/**
 * @param urlTitle The url_title
 */
public void setUrlTitle(String urlTitle) {
    this.urlTitle = urlTitle;
}

}

RequestInterface

import java.util.List;

import retrofit2.Call;
import retrofit2.http.GET;

/**
 * Created by Shweta.Chauhan on 13/07/16.
 */

public interface RequestInterface {

   @GET("facts/json/progress/all")
   Call<List<Datum>> getJSON();
}

DataAdapter

import android.content.Context;
import android.support.v7.widget.RecyclerView;
import android.view.LayoutInflater;
import android.view.View;
import android.view.ViewGroup;
import android.widget.TextView;

import java.util.ArrayList;
import java.util.List;

/**
 * Created by Shweta.Chauhan on 13/07/16.
 */

public class DataAdapter extends RecyclerView.Adapter<DataAdapter.MyViewHolder>{

private Context context;
private List<Datum> dataList;

public DataAdapter(Context context, List<Datum> dataList) {
    this.context = context;
    this.dataList = dataList;
}

@Override
public MyViewHolder onCreateViewHolder(ViewGroup parent, int viewType) {
    View view= LayoutInflater.from(parent.getContext()).inflate(R.layout.data,parent,false);
    return new MyViewHolder(view);
}

@Override
public void onBindViewHolder(MyViewHolder holder, int position) {
    holder.categoryTV.setText(dataList.get(position).getCategory());
    holder.urltitleTV.setText(dataList.get(position).getUrlTitle());

}

@Override
public int getItemCount() {
    return dataList.size();
}

public class MyViewHolder extends RecyclerView.ViewHolder{

    public TextView categoryTV, urltitleTV;

    public MyViewHolder(View itemView) {
        super(itemView);
        categoryTV = (TextView) itemView.findViewById(R.id.txt_category);
        urltitleTV = (TextView)     itemView.findViewById(R.id.txt_urltitle);
    }
}
}

और अंत में MainActivity.java

import android.support.v7.app.AppCompatActivity;
import android.os.Bundle;
import android.support.v7.widget.LinearLayoutManager;
import android.support.v7.widget.RecyclerView;
import android.util.Log;
import java.util.ArrayList;
import java.util.Arrays;
import java.util.List;

import retrofit2.Call;
import retrofit2.Callback;
import retrofit2.Response;
import retrofit2.Retrofit;
import retrofit2.converter.gson.GsonConverterFactory;

public class MainActivity extends AppCompatActivity {

private RecyclerView recyclerView;
private DataAdapter dataAdapter;
private List<Datum> dataArrayList;

@Override
protected void onCreate(Bundle savedInstanceState) {
    super.onCreate(savedInstanceState);
    setContentView(R.layout.activity_main);
    initViews();
}

private void initViews(){
    recyclerView=(RecyclerView) findViewById(R.id.recycler_view);
    recyclerView.setLayoutManager(new LinearLayoutManager(getApplicationContext()));
    loadJSON();
}

private void loadJSON(){
    dataArrayList = new ArrayList<>();
    Retrofit retrofit=new Retrofit.Builder().baseUrl("https://www.whitehouse.gov/").addConverterFactory(GsonConverterFactory.create()).build();
    RequestInterface requestInterface=retrofit.create(RequestInterface.class);
    Call<List<Datum>> call= requestInterface.getJSON();
    call.enqueue(new Callback<List<Datum>>() {
        @Override
        public void onResponse(Call<List<Datum>> call, Response<List<Datum>> response) {
            dataArrayList = response.body();
            dataAdapter=new DataAdapter(getApplicationContext(),dataArrayList);
            recyclerView.setAdapter(dataAdapter);
        }

        @Override
        public void onFailure(Call<List<Datum>> call, Throwable t) {
            Log.e("Error",t.getMessage());
        }
    });
}
}

5
इस तरह की समस्या के लिए सबसे अच्छा जवाब
निकी मनाली

4
यह पूरी तरह से विशेष रूप से रेट्रोफिट उपयोगकर्ताओं के लिए सवाल का जवाब देता है। स्पष्टता चाहने वाले किसी भी व्यक्ति के लिए, आपको जिस भाग की सबसे अधिक आवश्यकता है, वह है कॉल <सूची <डेटम >> गेटजोन ();
कार्लोस एनीोना

13

वैकल्पिक हो सकता है

अपनी प्रतिक्रिया देने के लिए

myCustom_JSONResponse

{"master":[
   {
      "updated_at":"2012-03-02 21:06:01",
      "fetched_at":"2012-03-02 21:28:37.728840",
      "description":null,
      "language":null,
      "title":"JOHN",
      "url":"http://rus.JOHN.JOHN/rss.php",
      "icon_url":null,
      "logo_url":null,
      "id":"4f4791da203d0c2d76000035",
      "modified":"2012-03-02 23:28:58.840076"
   },
   {
      "updated_at":"2012-03-02 14:07:44",
      "fetched_at":"2012-03-02 21:28:37.033108",
      "description":null,
      "language":null,
      "title":"PETER",
      "url":"http://PETER.PETER.lv/rss.php",
      "icon_url":null,
      "logo_url":null,
      "id":"4f476f61203d0c2d89000253",
      "modified":"2012-03-02 23:28:57.928001"
   }
]
}

के बजाय

server_JSONResponse

[
   {
      "updated_at":"2012-03-02 21:06:01",
      "fetched_at":"2012-03-02 21:28:37.728840",
      "description":null,
      "language":null,
      "title":"JOHN",
      "url":"http://rus.JOHN.JOHN/rss.php",
      "icon_url":null,
      "logo_url":null,
      "id":"4f4791da203d0c2d76000035",
      "modified":"2012-03-02 23:28:58.840076"
   },
   {
      "updated_at":"2012-03-02 14:07:44",
      "fetched_at":"2012-03-02 21:28:37.033108",
      "description":null,
      "language":null,
      "title":"PETER",
      "url":"http://PETER.PETER.lv/rss.php",
      "icon_url":null,
      "logo_url":null,
      "id":"4f476f61203d0c2d89000253",
      "modified":"2012-03-02 23:28:57.928001"
   }
]

कोड

  String server_JSONResponse =.... // the string in which you are getting your JSON Response after hitting URL
String myCustom_JSONResponse="";// in which we will keep our response after adding object element to it
     MyClass apiResponse = new MyClass();

     myCustom_JSONResponse="{\"master\":"+server_JSONResponse+"}";



    apiResponse = gson.fromJson(myCustom_JSONResponse, MyClass .class);

इसके बाद यह सिर्फ कोई और होगा GSON Parsing


क्या होगा अगर मैं अपना जसन प्रारूप नहीं बदल पा रहा हूँ? मैं अपने मॉडल वर्ग को स्थापित करने के लिए वॉली के गसन अनुरोध का उपयोग कर रहा हूं। यह कैसे करना है? धन्यवाद
Kaveesh कंवल

@KesheshKanwal इस सूत्र में दिए गए अन्य समाधानों को आज़माएं, इसके अलावा मुझे कोई पता नहीं है
DeltaCap019

8

GSON उपयोगकर्ता गाइड के अनुसार , आप नहीं कर सकते।

संग्रह की सीमाएँ

मनमानी वस्तुओं के संग्रह को क्रमबद्ध कर सकते हैं, लेकिन इससे विस्मरण नहीं कर सकते। क्योंकि उपयोगकर्ता के पास परिणामी वस्तु के प्रकार को इंगित करने का कोई तरीका नहीं है


7
उसके पास मनमानी वस्तुओं का संग्रह नहीं है, उसके पास एक विशिष्ट प्रकार की वस्तु का संग्रह है जो Gsonखुशी के साथ
निपटेगा

वास्तव में, मैंने TypeToken के साथ एक उत्तर लिखने की शुरुआत की थी जैसा आपने किया था, लेकिन चूंकि सामान्य प्रकार रनटाइम पर एम्बेडेड नहीं है, इसलिए मैंने यह नहीं देखा कि यह कैसे संभव है। (हालांकि मैंने इसका परीक्षण नहीं किया है)।
njzk2

3

यह एक Json सरणी सूची की तरह दिखता है ArrayList। डेटा को संभालने के लिए इसका सबसे अच्छा उपयोग करने के लिए। अपने एपीआई के अंत में इस तरह से सरणी सूची जोड़ें

 @GET("places/")
Call<ArrayList<Place>> getNearbyPlaces(@Query("latitude") String latitude, @Query("longitude") String longitude);

1

आपको नीचे दिए अनुसार अपनी प्रतिक्रिया के अतिरिक्त प्रकार के बारे में बताना चाहिए

import com.google.common.reflect.TypeToken;
import java.lang.reflect.Type;


Type collectionType = new TypeToken<List<UserSite>>(){}.getType();
List<UserSite> userSites  = gson.fromJson( response.getBody() , collectionType);

1

मुझे यकीन नहीं है कि यह GSON का उपयोग करने का सबसे अच्छा तरीका है, लेकिन मेरे लिए काम करता है। आप इस तरह से कुछ का उपयोग कर सकते हैं MainActivity:

 public void readJson() {
    dataArrayList = new ArrayList<>();
    String json = "[\n" + IOHelper.getData(this) + "\n]\n";
    Log.d(TAG, json);
    try{
        JSONArray channelSearchEnums = new JSONArray(json);

        for(int i=0; i< channelSearchEnums.length(); i++)
        {
            JSONObject enum = channelSearchEnums.getJSONObject(i);
            ChannelSearchEnum channel = new ChannelSearchEnum(
                   enum.getString("updated_at"), enum.getString("fetched_at"),
                   enum.getString("description"), enum.getString("language"),
                   enum.getString("title"), enum.getString("url"),
                   enum.getString("icon_url"), enum.getString("logo_url"),
                   enum.getString("id"), enum.getString("modified"))         

                   dataArrayList.add(channel);
        }

         //The code and place you want to show your data            

    }catch (Exception e)
    {
        Log.d(TAG, e.getLocalizedMessage());
    }
}

आपके पास केवल तार हैं, लेकिन यदि आपके पास युगल या इंट होंगे, तो आप डाल सकते हैं getDoubleया getIntभी कर सकते हैं।

IOHelperकक्षा का तरीका अगला है (यहां, आंतरिक संग्रहण पर पथ सहेजा गया है):

 public static String getData(Context context) {
    try {
        File f = new File(context.getFilesDir().getPath() + "/" + fileName);
        //check whether file exists
        FileInputStream is = new FileInputStream(f);
        int size = is.available();
        byte[] buffer = new byte[size];
        is.read(buffer);
        is.close();
        return new String(buffer);
    } catch (IOException e) {
        Log.e("TAG", "Error in Reading: " + e.getLocalizedMessage());
        return null;
    }
}

यदि आप इसके बारे में अधिक जानकारी चाहते हैं, तो आप इस वीडियो को देख सकते हैं , जहां मुझे कोड मिलता है readJson(); और यह धागा जहां मुझे कोड मिलता है getData()


0

Kotlin:

var list=ArrayList<Your class name>()
val listresult: Array<YOUR CLASS NAME> = Gson().fromJson(
                YOUR JSON RESPONSE IN STRING,
                Array<Your class name>:: class.java)

list.addAll(listresult)

मैं कुछ भी नहीं बढ़ा या घटाया, केवल संपादित किया।
शाम
हमारी साइट का प्रयोग करके, आप स्वीकार करते हैं कि आपने हमारी Cookie Policy और निजता नीति को पढ़ और समझा लिया है।
Licensed under cc by-sa 3.0 with attribution required.