Saturday, 12 September 2015

Android: Why, When and How to use AsyncTask with example

What is AsyncTask?
               AsyncTask is an abstract class provided by Android which helps us to use the UI thread properly. This class allows us to perform long/background operations and show its result on the UI thread without having to manipulate threads.
When to use AsyncTask?
            Android implements single thread model and whenever an android application is launched, a thread is created. Assuming we are doing network operation on a button click in our application. On button click a request would be made to the server and response will be awaited. Due to single thread model of android, till the time response is awaited our screen is non-responsive. So we should avoid performing long running operations on the UI thread. This includes file and network access.
            To overcome this we can create new thread and implement run method to perform this network call, so UI remains responsive.
            But since Android follows single thread model and Android UI toolkit is not thread safe, so if there is a need to make some change to the UI based on the result of the operation performed, then this approach may lead some issues.

So the Android framework has given a very good pattern which is enveloped into AsyncTask.

Note: AsyncTask should ideally be used for operations that take few seconds. Some tasks keep the thread running for long time so in that case it is recommended to use java.util.concurrent package such as Executor, ThreadPoolExecutor and FutureTask.

AsyncTask has four steps:

  1. doInBackgroundCode performing long running operation goes in this method.  When onClick method is executed on click of button, it calls execute method which accepts parameters and automatically calls doInBackground method with the parameters passed.
  2. onPostExecuteThis method is called after doInBackground method completes processing. Result from doInBackground is passed to this method.
  3. onPreExecuteThis method is called before doInBackground method is called.
  4. onProgressUpdate: This method is invoked by calling publishProgress anytime from doInBackground call this method.
The task can be cancelled by invoking cancel(boolean) method. This will cause subsequent calls to isCancelled()to return true. After invoking this method, onCancelled(Object) method is called instead of onPostExecute() after doInBackground() returns. 

How to use AyncTask? Example application

         In this sample application I just made the process to sleep for some period of time instead of doing network operation.( Just to explain the concept of AsyncTask. This is not a realtime application).
In the UI thread user enters a value (time in milli seconds) which will be passed to AsyncTaskRunner.
AsyncTaskRunner is a private class which extends AsyncTask.

Steps to create the application:

1. Create new android application project in eclipse with default settings.
2. Create the layout for the application.
3. Create the MainActivity and AsyncTaskRunner at com.example.asynctaskdemo.MainActivity.java
4. Run the application as given in the below screenshots.

Activity_Asynctask.xml
   <RelativeLayout xmlns:android="http://schemas.android.com/apk/res/android"
    xmlns:tools="http://schemas.android.com/tools"
    android:layout_width="match_parent"
    android:layout_height="match_parent"
    tools:context=".MainActivity" >
 
     <TextView
        android:id="@+id/tv_time"
        android:layout_width="wrap_content"
        android:layout_height="wrap_content"
        android:textSize="10pt"
        android:textColor="#444444"
        android:layout_alignParentLeft="true"
        android:layout_marginRight="9dip"
        android:layout_marginTop="20dip"
        android:layout_marginLeft="10dip"
        android:text="Sleep time in milliseconds:"/>
    <EditText
        android:id="@+id/et_time"
        android:layout_width="150dip"
        android:layout_height="wrap_content"
        android:background="@android:drawable/editbox_background"
        android:layout_toRightOf="@id/tv_time"
        android:layout_alignTop="@id/tv_time"
        android:inputType="text"
        />

    <Button
        android:id="@+id/btn_do_it"
        android:layout_width="fill_parent"
        android:layout_height="wrap_content"
        android:layout_alignLeft="@+id/tv_time"
        android:layout_below="@id/et_time"
        android:layout_marginTop="15dip"
        android:text="Run Async task" />

     <TextView
         android:id="@+id/tv_result"
         android:layout_width="fill_parent"
         android:layout_height="100dip"
         android:layout_alignLeft="@+id/btn_do_it"
         android:layout_below="@id/btn_do_it"
         android:layout_marginRight="9dip"
         android:layout_marginTop="15dip"
         android:text=""
         android:textColor="#AA0000"
         android:textSize="7pt" />

</RelativeLayout>

AsyncTaskActivity.Java

package com.example.asynctask;

import android.app.Activity;
import android.os.AsyncTask;
import android.os.Bundle;
import android.view.Menu;
import android.view.MenuItem;
import android.view.View;
import android.widget.Button;
import android.widget.EditText;
import android.widget.TextView;

public class AsyncTaskActivity extends Activity {
private Button button;
private EditText time;
private TextView finalResult;

@Override
protected void onCreate(Bundle savedInstanceState) {
 super.onCreate(savedInstanceState);
 setContentView(R.layout.activity_async_task);
 time = (EditText) findViewById(R.id.et_time);
 button = (Button) findViewById(R.id.btn_do_it);
 finalResult = (TextView) findViewById(R.id.tv_result);
 button.setOnClickListener(new View.OnClickListener() {
  @Override
  public void onClick(View v) {
   AsyncTaskRunner runner = new AsyncTaskRunner();
   String sleepTime = time.getText().toString();
   runner.execute(sleepTime);
  }
 });
}

 
/**
 * @author Raj
 * Private class which runs the long operation. ( Sleeping for some time )
 */
private class AsyncTaskRunner extends AsyncTask<String, String, String> {

 private String resp;

 @Override
 protected String doInBackground(String... params) {
  publishProgress("Sleeping..."); // Calls onProgressUpdate()
  try {
   // Do your long operations here and return the result
   int time = Integer.parseInt(params[0]);    
   // Sleeping for given time period
   Thread.sleep(time);
   resp = "Slept for " + time + " milliseconds";
  } catch (InterruptedException e) {
   e.printStackTrace();
   resp = e.getMessage();
  } catch (Exception e) {
   e.printStackTrace();
   resp = e.getMessage();
  }
  return resp;
 }

 /*
  * (non-Javadoc)
  * 
  * @see android.os.AsyncTask#onPostExecute(java.lang.Object)
  */
 @Override
 protected void onPostExecute(String result) {
  // execution of result of Long time consuming operation
  finalResult.setText(result);
 }

 /*
  * (non-Javadoc)
  * 
  * @see android.os.AsyncTask#onPreExecute()
  */
 @Override
 protected void onPreExecute() {
  // Things to be done before execution of long running operation. For
  // example showing ProgessDialog
 }

 /*
  * (non-Javadoc)
  * 
  * @see android.os.AsyncTask#onProgressUpdate(Progress[])
  */
 @Override
 protected void onProgressUpdate(String... text) {
  finalResult.setText(text[0]);
  // Things to be done while execution of long running operation is in
  // progress. For example updating ProgessDialog
 }
}
}

OutPut




No comments:

Post a Comment

Ads Inside Post