Tuesday, 31 December 2013

Android Background Processing with Handlers and AsyncTask and Loaders


Welcome Friends !!

Most of the time it is very difficult to understand how we are going to handle the async task in application development. The heavy task like downloading a image from the internet or other networking task should not be done from the main thread.

 I am going to write an article that describes the usage of Threads, Handlers and AsyncTask in your application. It also covers how to handle the application lifecycle together with threads. It is based on Eclipse 4.2, Java 1.6 and Android 4.2.

1. Android's user interface thread

1.1. Main thread

Android modifies the user interface and handles input events from one single user interface thread. This thread is also called the main thread.
Android collects all events in a queue and processed an instance of the Looper class.
                          Message Queue in Android with Looper

1.2. Why using concurrency?

If the programmer does not use any concurrency constructs, all code of an Android application runs in the main thread and every statement is executed after each other.
If you perform a long lasting operation, for example accessing data from the Internet, the application blocks until the corresponding operation has finished.
To provide a good user experience all potentially slow running operations in an Android application should run asynchronously, e.g. via some way of concurrency constructs of the Java language or the Android framework. This includes all potential slow operations, like network, file and database access and complex calculations.

2. Using Java threads in Android

2.1. Using Java threading in Android

Android supports the usage of the Thread class to perform asynchronous processing.
Android also supplies the java.util.concurrent package to perform something in the background, e.g. using theThreadPools and Executor classes.
If you need to update the user interface from a new Thread, you need to synchronize with the user interface thread.

2.2. Disadvantages of using Java threads in Android

If you use Java threads you have to handle the following requirements
in your own code:
  • Synchronization with the main thread if you post back results to the user interface
  • No default for canceling the thread
  • No default thread pooling
  • No default for handling configuration changes in Android

3. Concurrency constructs in Android

Android provides additional constructs to handle concurrently in comparison with standard Java. You can use theandroid.os.Handler class or the AsyncTasks classes. More sophisticated approach are based on the Loader class, retained Fragments and services.

4. Handler

4.1. Purpose of the Handler class

The Handler class can be used to register to a thread and provides a simple channel to send data to this thread.
Handler object registers itself with the thread in which it is created. For example, if you create a new instance of the Handler class in the onCreate() method of your activity, the resulting Handler object can be used to post data to the main thread.
The data which can be posted via the Handler class can be an instance of the Message or the Runnable class.


4.2. Creating and reusing instances of Handlers

To use a handler you have to subclass it and override the handleMessage() method to process messages.
Your thread can post messages via the sendMessage(Message) method or via the sendEmptyMessage() method to theHandler object.
To process a Runnable you can use the post() method.
To avoid object creation you can also reuse the existing Handler object of your activity.
// Reuse existing handler if you don't 
// have to override the message processing
handler = getWindow().getDecorView().getHandler(); 
The View class allows you to post objects of type Runnable via the post() method.

4.3. Example

The following code demonstrates the usage of a Handler via a View.
Assume your activity uses the following layout.
<?xml version="1.0" encoding="utf-8"?>
<LinearLayout xmlns:android="http://schemas.android.com/apk/res/android"
    android:layout_width="match_parent"
    android:layout_height="match_parent"
    android:orientation="vertical" >

    <ProgressBar
        android:id="@+id/progressBar1"
        style="?android:attr/progressBarStyleHorizontal"
        android:layout_width="match_parent"
        android:layout_height="wrap_content"
        android:indeterminate="false"
        android:max="10"
        android:padding="4dip" >
    </ProgressBar>
    
   <TextView
        android:id="@+id/textView1"
        android:layout_width="wrap_content"
        android:layout_height="wrap_content"
        android:text="" >
      </TextView>
    <Button
        android:id="@+id/button1"
        android:layout_width="wrap_content"
        android:layout_height="wrap_content"
        android:onClick="startProgress"
        android:text="Start Progress" >
    </Button>

</LinearLayout> 
With the following the ProgressBar get updated once the users presses the Button.
import android.app.Activity;
import android.os.Bundle;
import android.view.View;
import android.widget.ProgressBar;
import android.widget.TextView;

public class ProgressTestActivity extends Activity {
  private ProgressBar progress;
  private TextView text;

  @Override
  public void onCreate(Bundle savedInstanceState) {
    super.onCreate(savedInstanceState);
    setContentView(R.layout.main);
    progress = (ProgressBar) findViewById(R.id.progressBar1);
    text = (TextView) findViewById(R.id.textView1);

  }

  public void startProgress(View view) {
    // do something long
    Runnable runnable = new Runnable() {
      @Override
      public void run() {
        for (int i = 0; i <= 10; i++) {
          final int value = i;
           doFakeWork();
          progress.post(new Runnable() {
            @Override
            public void run() {
              text.setText("Updating");
              progress.setProgress(value);
            }
          });
        }
      }
    };
    new Thread(runnable).start();
  }

  // Simulating something timeconsuming
  private void doFakeWork() {
    try {
      Thread.sleep(2000);
    } catch (InterruptedException e) {
      e.printStackTrace();
    }
  }

} 

5. AsyncTask

5.1. Purpose of the AsyncTask class

The AsyncTask class encapsulates the creation of a background process and the synchronization with the main thread. It also supports reporting progress of the running tasks.

5.2. Using the AsyncTask class

To use AsyncTask you must subclass it. AsyncTask uses generics and varargs. The parameters are the following AsyncTask <TypeOfVarArgParams , ProgressValue , ResultValue> .
An AsyncTask is started via the execute() method.
The execute() method calls the doInBackground() and the onPostExecute() method.
TypeOfVarArgParams is passed into the doInBackground() method as input, ProgressValue is used for progress information and ResultValue must be returned from doInBackground() method and is passed to onPostExecute() as a parameter.
The doInBackground() method contains the coding instruction which should be performed in a background thread. This method runs automatically in a separate Thread.
The onPostExecute() method synchronizes itself again with the user interface thread and allows it to be updated. This method is called by the framework once the doInBackground() method finishes.

5.3. Parallel execution of several AsyncTasks

Android executes AsyncTask tasks before Android 1.6 and again as of Android 3.0 in sequence by default.
You can tell Android to run it in parallel with the usage of the executeOnExecutor() method, specifyingAsyncTask.THREAD_POOL_EXECUTOR as first parameter.
The following code snippet demonstrates that.
// ImageLoader extends AsyncTask
ImageLoader imageLoader = new ImageLoader(imageView);

// Execute in parallel
imageLoader.executeOnExecutor(AsyncTask.THREAD_POOL_EXECUTOR, "http://url.com/image.png"); 

5.4. Disadvantages of using AsyncTasks

The AsyncTask does not handle configuration changes automatically, i.e. if the activity is recreated, the programmer has to handle that in his coding.
A common solution to this is to declare the AsyncTask in a retained headless fragment.

5.5. Example: AsyncTask

The following code demonstrates how to use the AsyncTask class to download the content of a webpage.
Create a new Android project called de.vogella.android.asynctask with an activity called ReadWebpageAsyncTask. Add the android.permission.INTERNET permission to your AndroidManifest.xml file.
Create the following layout.
<?xml version="1.0" encoding="utf-8"?>
<LinearLayout xmlns:android="http://schemas.android.com/apk/res/android"
    android:layout_width="match_parent"
    android:layout_height="match_parent"
    android:orientation="vertical" >

    <Button
        android:id="@+id/readWebpage"
        android:layout_width="match_parent"
        android:layout_height="wrap_content"
        android:onClick="onClick"
        android:text="Load Webpage" >
    </Button>

    <TextView
        android:id="@+id/TextView01"
        android:layout_width="match_parent"
        android:layout_height="match_parent"
        android:text="Placeholder" >
    </TextView>

</LinearLayout> 
Change your activity to the following:
import java.io.BufferedReader;
import java.io.InputStream;
import java.io.InputStreamReader;

import org.apache.http.HttpResponse;
import org.apache.http.client.methods.HttpGet;
import org.apache.http.impl.client.DefaultHttpClient;

import de.vogella.android.asyntask.R;

import android.app.Activity;
import android.os.AsyncTask;
import android.os.Bundle;
import android.view.View;
import android.widget.TextView;

public class ReadWebpageAsyncTask extends Activity {
  private TextView textView;

  @Override
  public void onCreate(Bundle savedInstanceState) {
    super.onCreate(savedInstanceState);
    setContentView(R.layout.main);
    textView = (TextView) findViewById(R.id.TextView01);
  }

  private class DownloadWebPageTask extends AsyncTask<String, Void, String> {
    @Override
    protected String doInBackground(String... urls) {
      String response = "";
      for (String url : urls) {
        DefaultHttpClient client = new DefaultHttpClient();
        HttpGet httpGet = new HttpGet(url);
        try {
          HttpResponse execute = client.execute(httpGet);
          InputStream content = execute.getEntity().getContent();

          BufferedReader buffer = new BufferedReader(new InputStreamReader(content));
          String s = "";
          while ((s = buffer.readLine()) != null) {
            response += s;
          }

        } catch (Exception e) {
          e.printStackTrace();
        }
      }
      return response;
    }

    @Override
    protected void onPostExecute(String result) {
      textView.setText(result);
    }
  }

  public void onClick(View view) {
    DownloadWebPageTask task = new DownloadWebPageTask();
    task.execute(new String[] { "http://www.vogella.com" });

  }
} 
If you run your application and press your button then the content of the defined webpage is read in the background. Once this process is done your TextView is updated.

Monday, 30 December 2013

Android : Weather application JSON data parsing


Welcome Guys !! 

Here I am going to write the article about how to parse the JSON data that is retrieved from the API service provider.
I am developing a simple weather application and i am using the http://api.openweathermap.org service.
This service return JSON data of current weather information.

Queries for data can be sent on the following URL:


private static String BASE_URL = "http://api.openweathermap.org/data/2.5/weather?q=";

We are going to attach a query parameter like "Rome,IT" and it will return me the all required information :
The URL will look like :

                        http://api.openweathermap.org/data/2.5/weather?q=Rome,IT
We can cheak the JSON using the above URL
The formatted readable data can be found by copying the raw browser data and using .JSON formattor http://jsonformatter.curiousconcept.com.

{
   "coord":{
      "lon":12.5,
      "lat":41.9
   },
   "sys":{
      "message":0.1019,
      "country":"Italy",
      "sunrise":1388385449,
      "sunset":1388418491
   },
   "weather":[
      {
         "id":500,
         "main":"Rain",
         "description":"light rain",
         "icon":"10n"
      },
      {
         "id":701,
         "main":"Mist",
         "description":"mist",
         "icon":"50n"
      }
   ],
   "base":"gdps stations",
   "main":{
      "temp":281.26,
      "pressure":1019,
      "humidity":93,
      "temp_min":280.93,
      "temp_max":282.15
   },
   "wind":{
      "speed":2.6,
      "deg":30,
      "var_beg":360,
      "var_end":70
   },
   "rain":{
      "1h":0.25
   },
   "clouds":{
      "all":75
   },
   "dt":1388440902,
   "id":3169070,
   "name":"Rome",
   "cod":200
}



Step1: First we are going to created a model class that will hold the each instance of parsed JSON info.

public class Weather {
double lon=0.0;
double lang=0.0;
String country="";
long sunrise=0;
long sunset=0;
String main="";
String description="";
String icon="";
double temp=0.0;
double pressure=0.0;
double humidity=0.0;
double temp_min=0.0;
double temp_max=0.0;
double speed=0.0;
double deg=0.0;
double cloud=0.0;
String name="";

}



Step 2: We are going to start a ASYNC task that will send the request to the server demanding the JSON.

public class JSONWeatherTask extends AsyncTask<String, Integer, String>{

@Override
protected String doInBackground(String... arg0) {
// TODO Auto-generated method stub
Weather weather = new Weather();
   String data = ( (new WeatherHttpClient()).getWeatherData(arg0[0]));
   System.out.println(data);

   try {
       weather = JSONWeatherParser.getWeather(data);
       System.out.println(weather.lon);
       System.out.println(weather.lang);
       System.out.println(weather.country);
       System.out.println(weather.sunrise);
       System.out.println(weather.sunset);
       System.out.println(weather.main);
       System.out.println(weather.description);
   
     
   } catch (JSONException e) {              
       e.printStackTrace();
   }
 
   return data;
}

}

Step 3:  We are going to establish a HTTP URL Connection and read the required data returned using the input stream readers and store them in a String type.
This data in string form will be parsed by a simple Json Parser and required results will be displayed.

public class WeatherHttpClient {

private static String BASE_URL = "http://api.openweathermap.org/data/2.5/weather?q=";
    private static String IMG_URL = "http://openweathermap.org/img/w/";

   
    public String getWeatherData(String location) {
        HttpURLConnection con = null ;
        InputStream is = null;

        try {
            con = (HttpURLConnection) ( new URL(BASE_URL + location)).openConnection();
            con.setRequestMethod("GET");
            con.setDoInput(true);
            con.setDoOutput(true);
            con.connect();
           
            // Let's read the response
            StringBuffer buffer = new StringBuffer();
            is = con.getInputStream();
            BufferedReader br = new BufferedReader(new InputStreamReader(is));
            String line = null;
            while (  (line = br.readLine()) != null )
                buffer.append(line + "\r\n");
           
            is.close();
            con.disconnect();
            return buffer.toString();
        }
        catch(Throwable t) {
            t.printStackTrace();
        }
        finally {
            try { is.close(); } catch(Throwable t) {}
            try { con.disconnect(); } catch(Throwable t) {}
        }

        return null;
               
    }

}


Step 4: In the last we are going to parse result data retrieved and display this in our application.

NOTE : While updating the User Interface , always use a separate thread. The separate thread can be created using 
runOnUiThread(new Runnable() {
                public void run() {
                                  }
                                                 }

public class JSONWeatherParser {
// We start extracting the info


private static JSONObject getObject(String tagName, JSONObject jObj)  throws JSONException {
   JSONObject subObj = jObj.getJSONObject(tagName);
   return subObj;
}

private static String getString(String tagName, JSONObject jObj) throws JSONException {
   return jObj.getString(tagName);
}

private static float  getFloat(String tagName, JSONObject jObj) throws JSONException {
   return (float) jObj.getDouble(tagName);
}

private static int  getInt(String tagName, JSONObject jObj) throws JSONException {
   return jObj.getInt(tagName);
}

public static Weather getWeather(String data) throws JSONException {
// TODO Auto-generated method stub
// We create out JSONObject from the data
Weather weather=new Weather();
JSONObject jObj = new JSONObject(data);
JSONObject coordObj = getObject("coord", jObj);
weather.lon=getFloat("lon", coordObj);
weather.lang=getFloat("lon", coordObj);
JSONObject sysObj = getObject("sys", jObj);
weather.country=getString("country", sysObj);
weather.sunrise=getInt("sunrise", sysObj);
weather.sunset=getInt("sunset", sysObj);
//We get weather info (This is an array)
JSONArray jArr = jObj.getJSONArray("weather");

// We use only the first value
JSONObject JSONWeather = jArr.getJSONObject(0);
weather.main=getString("main", JSONWeather);
weather.description=getString("description", JSONWeather);




return weather;
}



}


Note : Guys ... I have just written a sample parser which is quite simple and give you a  basic idea how things work.
The post will be updated later with the entire application to download.
For Any further question or actual source code for the application ... 

mail me at : ajayjaiswalhi@gmail.com


Thank You !!






Android : Effective and very accurate way to find the distance nearest distance in between two provided latitudes and longitudes


Android : Effective and very accurate way to find the distance nearest distance in between two provided latitudes and longitudes

We people while developing a Google Map API based application often come accross the problem to find the distance in between two provided latitude and longitude.
Here is a very effective algorithm to fin out the normal and very accurate distance in between them:

public static double distFrom(double lat1,double lng1,double lat2,double lng2) {
   double earthRadius = 3958.75;
   double dLat = Math.toRadians(lat2-lat1);
   double dLng = Math.toRadians(lng2-lng1);
   double a = Math.sin(dLat/2) * Math.sin(dLat/2) +
              Math.cos(Math.toRadians(lat1)) * Math.cos(Math.toRadians(lat2)) *
              Math.sin(dLng/2) * Math.sin(dLng/2);
   double c = 2 * Math.atan2(Math.sqrt(a), Math.sqrt(1-a));
   double dist = earthRadius * c;
   return dist;

   }


I have used this in my application and this is very accurate friends.

Thank you !!