Logo Questions Linux Laravel Mysql Ubuntu Git Menu
 

Android Parse JSON stuck on get task

I am trying to parse some JSON data. My code was working for awhile, and I am not sure what I changed to suddenly break the code. When I run my code I am not receiving any runtime errors or warnings. I create a new AsyncTask and execute this. When I call .get() on this new task, the debugger stalls on this line and takes more than 30 minutes. I have not been able to get the debugger or during run to complete this task.

JSON:

protected void setUp(Context context) {
    _context = context;
    getConfig();
}

// get config file
protected void getConfig() { 
    if (config != null)
        return;

    config = new Config();

    String url = configURL;
    AsyncTask<String, Integer, JSONObject> jsonTask = new DownloadJSONTask()
            .execute(url);
    JSONObject configItem = null;
    try {
        configItem = jsonTask.get(); //debugger pauses here
        if (configItem == null)
            return;
        config.configVersion = configItem.getString("field_configversion");
        config.currentAppVersion = configItem
                .getString("field_currentappversion");
        config.getSupportURL = configItem.getString("field_getsupporturl");
        config.getCatalogURL = configItem.getString("field_getcatalogurl");
        config.getDataVersion = configItem.getString("field_dataversion");
        config.getDataUrl = configItem.getString("field_dataurl");
        config.getDataApiKey = configItem.getString("field_dataapikey");
    } catch (InterruptedException e) {
        e.printStackTrace();
        System.err.println("Download of config interrupted");
    } catch (ExecutionException e) {
        e.printStackTrace();
        System.err.println("Download of config failed to execute");
    } catch (JSONException e) {
        e.printStackTrace();
    } 

    cacheStaticData(_context);
}

DownloadJSONTask.java

package com.example.simplegraph;

import java.io.BufferedReader;
import java.io.IOException;
import java.io.InputStream;
import java.io.InputStreamReader;

import org.apache.http.HttpEntity;
import org.apache.http.HttpResponse;
import org.apache.http.client.ClientProtocolException;
import org.apache.http.client.HttpClient;
import org.apache.http.client.methods.HttpGet;
import org.apache.http.impl.client.DefaultHttpClient;
import org.json.JSONException;
import org.json.JSONObject;

import android.content.Context;
import android.os.AsyncTask;

public class DownloadJSONTask extends AsyncTask<String, Integer, JSONObject> {
private HttpClient client = new DefaultHttpClient();
private HttpGet request;
private HttpResponse response;

DownloadJSONTask() {
    super();
}

// tries to grab the data for a JSONObject
@Override
protected JSONObject doInBackground(String... urls) {

    request = new HttpGet(urls[0]);
    try {
        response = client.execute(request);
        HttpEntity entity = response.getEntity();
        if (entity != null) {
            InputStream instream = entity.getContent();
            String result = convertStreamToString(instream);
            JSONObject json = new JSONObject(result);
            instream.close();
            return json;
        }
    } catch (ClientProtocolException e) {
        e.printStackTrace();
    } catch (IOException e) {
        e.printStackTrace();
    } catch (JSONException e) {
        e.printStackTrace();
    }
    return null;
}

// converts the InputStream to a string and add nl
private String convertStreamToString(InputStream is) {
    BufferedReader br = new BufferedReader(new InputStreamReader(is));
    StringBuilder sb = new StringBuilder();
    String line = null;
    try {
        while ((line = br.readLine()) != null) {
            sb.append(line + "\n");
        }
    } catch (IOException ioe) {
        ioe.printStackTrace();
    } finally {
        try {
            is.close();
        } catch (IOException ioe) {
            ioe.printStackTrace();
        }
    }
    return sb.toString();
}
}

And HomeActivity.java

@Override
public void onCreate(Bundle savedInstanceState) {
    super.onCreate(savedInstanceState);
    setContentView(R.layout.activity_home);

    new AddStringTask().execute();

}

class AddStringTask extends AsyncTask<Void, String, Void> {
    @Override
    protected Void doInBackground(Void... unused) {

        app = (EconApplication) getApplication();
        getApp().setUp(HomeActivity.this);
        HomeActivity.this.setUpDrawer();
        return (null);
    }

    @Override
    protected void onPostExecute(Void unused) {
        setUpDataDisplay();
        setUpGraphRange();
        createTable();
        createGraph(-1);
    }
}

QUESTION: Why is my code getting stuck on .get()?

like image 798
buczek Avatar asked Aug 08 '13 14:08

buczek


2 Answers

AsyncTask.get() blocks the caller thread. Use AsyncTask.execute() instead.

public final Result get ()

Added in API level 3

Waits if necessary for the computation to complete, and then retrieves its result.

Returns The computed result.

Drawing from

How do I return a boolean from AsyncTask?

Try the below

  new DownloadJSONTask(ActivityName.this).execute(url);

In your DownloadJSONTask

In the construcotr

    TheInterface listener;
    public DownloadJSONTask(Context context)
{

    listener = (TheInterface) context;

}

Interface

   public interface TheInterface {

    public void theMethod(ArrayList<String> result); // your result type

     }

In your doInbackground return the result. I am assuming its ArrayList of type String. Change the arraylist to what suits your requirement.

In your onPostExecute

   if (listener != null) 
   {
      listener.theMethod(result); // result is the ArrayList<String>
      // result returned in doInbackground 
      // result of doInbackground computation is a parameter to onPostExecute 
   }

In your activity class implement the interface

 public class ActivityName implements DownloadJSONTask.TheInterface

Then

 @Override
 public void theMethod(ArrayList<String> result) { // change the type of result according yo your requirement
 // use the arraylist here
 }

Edit: Alternative

You can makes your asynctask an inner class of your activity class. The result on doInbackground computation is a parameter to onPostExecute. Return result in doInbackground. Update ui in onPostExecute.

like image 115
Raghunandan Avatar answered Nov 17 '22 11:11

Raghunandan


You can greatly simplify everything using the droidQuery library:

$.getJSON("http://www.example.com", null, new Function() {//this will run using an AsyncTask, get the JSON, and return either a JSONObject or JSONArray on the UI Thread.
    @Overrde
    public void invoke($ droidQuery, Object... params) {
        if (params[0] instanceof JSONObject) { //it's often ok just to assume a JSONObject, making your first line simply: JSONObject obj = (JSONObject) params[0];
            //JSONObject is returned
            JSONObject json = (JSONObject) params[0];
            //to easily parse this Object, convert it to a map first:
            Map<String, ?> map = $.map(json);
            //then you can just make a call like this:
            if (map.contains("field_currentappversion")) {
                config.currentAppVersion = (String) map.get("field_currentappversion");
            }
        }
        else {
            //JSONArray is returned
            JSONArray json = (JSONArray) params[0];
            //if you got an array, you can easily convert it to an Object[] for parsing:
            Object[] array = $.makeArray(json);
        }
    }
});
like image 35
Phil Avatar answered Nov 17 '22 10:11

Phil