Satyen Udeshi
Satyen Udeshi

Reputation: 3243

Achieving Concurrency using AsyncTask?

I am developing an Android application to download images from my web server. All the Code is running fine. I am using Asynctask to download the images to my sdcard.

I am on a 4mbps connection but my application is taking about 8 mins to download 3 images (2.5 MB). I have read else where that Asynctask automatically manages Thread creation, so now what I can do to achieve concurrency ?

I am posting my code below. The code Below is for my Asynctask activity that downloads the image from server to sdcard.

public class BitmapDownloaderTask extends AsyncTask<String, Void, Bitmap> {
private String url;
Bitmap bitmap1;
String sdCard;
private final WeakReference<ImageView> imageViewReference;

public BitmapDownloaderTask(ImageView imageView) {
    imageViewReference = new WeakReference<ImageView>(imageView);
}

@Override
// Actual download method, run in the task thread
protected Bitmap doInBackground(String... params) {

    // params comes from the execute() call: params[0] is the url.
    bitmap1 = downloadBitmap(params[0]);

    boolean avail = isMemorySizeAvailableAndroid(bitmap1.getRowBytes(),
            Environment.isExternalStorageEmulated());
    if (avail) {
        try {
            sdCard = Environment.getExternalStorageDirectory().toString()
                    + "/MyCatalogue";
            File f1 = new File(sdCard);
            if (!f1.exists()) {
                f1.mkdirs();
            }
            String filename1 = params[0].substring(params[0]
                    .lastIndexOf("/") + 1);
            File file1 = new File(f1.toString(), filename1);

            OutputStream stream1 = new FileOutputStream(file1);
            bitmap1.compress(CompressFormat.JPEG, 100, stream1);
            Log.w("Abhishek", "card is " + sdCard);
        } catch (Exception e) {
            e.printStackTrace();

        }
    }
    Log.w("ImageDownloader", "Success bitmap is" + bitmap1);
    return downloadBitmap(params[0]);
}

protected static boolean isMemorySizeAvailableAndroid(long download_bytes,
        boolean isExternalMemory) {
    boolean isMemoryAvailable = false;
    long freeSpace = 0;

    // if isExternalMemory get true to calculate external SD card available
    // size
    if (isExternalMemory) {
        try {
            StatFs stat = new StatFs(Environment
                    .getExternalStorageDirectory().getPath());
            freeSpace = (long) stat.getAvailableBlocks()
                    * (long) stat.getBlockSize();
            if (freeSpace > download_bytes) {
                isMemoryAvailable = true;
            } else {
                isMemoryAvailable = false;
            }
        } catch (Exception e) {
            e.printStackTrace();
            isMemoryAvailable = false;
        }
    } else {
        // find phone available size
        try {
            StatFs stat = new StatFs(Environment.getDataDirectory()
                    .getPath());
            freeSpace = (long) stat.getAvailableBlocks()
                    * (long) stat.getBlockSize();
            if (freeSpace > download_bytes) {
                isMemoryAvailable = true;
            } else {
                isMemoryAvailable = false;
            }
        } catch (Exception e) {
            e.printStackTrace();
            isMemoryAvailable = false;
        }
    }

    return isMemoryAvailable;
}

@Override
// Once the image is downloaded, associates it to the imageView
protected void onPostExecute(Bitmap bitmap) {
    if (isCancelled()) {
        bitmap = null;
    }

    if (imageViewReference != null) {
        ImageView imageView = imageViewReference.get();
        if (imageView != null) {
            imageView.setImageBitmap(bitmap);
        }
    }
}

static Bitmap downloadBitmap(String url) {
    final AndroidHttpClient client = AndroidHttpClient
            .newInstance("Android");
    final HttpGet getRequest = new HttpGet(url);

    try {
        HttpResponse response = client.execute(getRequest);
        final int statusCode = response.getStatusLine().getStatusCode();
        if (statusCode != HttpStatus.SC_OK) {
            Log.w("ImageDownloader", "Error " + statusCode
                    + " while retrieving bitmap from " + url);
            return null;
        } else {
            Log.w("ImageDownloader", "Success " + statusCode
                    + " while retrieving bitmap from " + url);
        }

        final HttpEntity entity = response.getEntity();
        if (entity != null) {
            InputStream inputStream = null;
            try {
                inputStream = entity.getContent();
                final Bitmap bitmap = BitmapFactory
                        .decodeStream(inputStream);
                return bitmap;
            } finally {
                if (inputStream != null) {
                    inputStream.close();
                }
                entity.consumeContent();
            }
        }
    } catch (Exception e) {
        // Could provide a more explicit error message for IOException or
        // IllegalStateException
        getRequest.abort();
        Log.w("ImageDownloader", "Error while retrieving bitmap from "
                + url);
    } finally {
        if (client != null) {
            client.close();
        }
    }
    return null;
}
}

Upvotes: 1

Views: 471

Answers (2)

Raghunandan
Raghunandan

Reputation: 133560

Use a executeOnExecutor

http://developer.android.com/reference/java/util/concurrent/Executor.html

 new BitmapDownloaderTask.executeOnExecutor(AsyncTask.THREAD_POOL_EXECUTOR, "your urls"); 

Quoting from docs

http://developer.android.com/reference/android/os/AsyncTask.html

When first introduced, AsyncTasks were executed serially on a single background thread. Starting with DONUT, this was changed to a pool of threads allowing multiple tasks to operate in parallel. Starting with HONEYCOMB, tasks are executed on a single thread to avoid common application errors caused by parallel execution.

If you truly want parallel execution, you can invoke executeOnExecutor(java.util.concurrent.Executor, Object[]) with THREAD_POOL_EXECUTOR.

Upvotes: 0

Dong Xiabin
Dong Xiabin

Reputation: 171

Why you download image twice in doInBackground() at the start and the end? You can return the bitmap just downloaded directly.

if your min sdk level >= 11, you can call executeOnExecutor of AsyncTask with param " THREAD_POOL_EXECUTOR" for concurrency.

if your min sdk level < 11, you can implements AsyncTask new API by reference the source code of AsyncTask.

Upvotes: 1

Related Questions