How to download multiple files concurrently using intentservice in Android?

Alston picture Alston · Sep 4, 2014 · Viewed 11.1k times · Source

I want to create a service similar to this one, (reference from Here), to download multiple files asynchronously in Android.

public static class DownloadingService extends IntentService {
public static String PROGRESS_UPDATE_ACTION = DownloadingService.class
        .getName() + ".newDownloadTask";
private ExecutorService mExec;
private CompletionService<NoResultType> mEcs;
private LocalBroadcastManager mBroadcastManager;
private List<DownloadTask> mTasks;

public DownloadingService() {
    super("DownloadingService");
    mExec = Executors.newFixedThreadPool( 3 ); // The reason to use multiple thread is to download files asynchronously. 
    mEcs = new ExecutorCompletionService<NoResultType>(mExec);
}

@Override
public int onStartCommand(Intent intent, int flags, int startId) {
    return super.onStartCommand(intent, flags, startId);
}

@Override
protected void onHandleIntent(Intent intent) {

  while(true)
  {
    if( cursor <= totalDownloadQueue.size() -1) {   //totalDownloadQueue is a static ArrayList which contains numerous DownloadTask
        mEcs.submit(totalDownloadQueue.get(cursor)); 
        cursor++; // The variable cursor is also a static int variable. 
    }
  }// continuously observing the totalDownloadQueue. If any download item is added. Then the thread are dispatched to handle that.
    mExec.shutdown();
}

The user can select download items among listview in different fragments. My strategy is that as the user select the items and press download button, these items are passed into DownloadTask which is responsible for downloading a file. Then the download tasks are added into the totalDownloadQueue.

Here are some questions:

  1. I know the intentservice is triggered by some defined action. But what I want to is to create a background service watching the totalDownloadQueue, if any new downloadtask is availabe, then some thread are called to operate the tasks.

    What's the side effect if I did so for this customized intentservice?

    What alternative class should I use? Please provide sample code along with the explanation, thanks.

  2. As I know, the initialization of the threads is only called by once. If I start the service at the beginning of the application and the threads should be killed as the user terminate the app.(I mean when he swipe out the window.) Do the threads exist after the user exit the application?

  3. If this approach still can't resolve the issue about downloading files asynchronously? What other strategy should I adopt? Please provide some example code or reference so that I can modify on it.

I have spent 7 days dealing with the complex requirement, any help please!

Answer

mmlooloo picture mmlooloo · Sep 6, 2014

to download multiple files asynchronously in Android.

and also i think you want to download simultaneously.

i think you misused the intentservice. intentservice has a looper and a handler and each call to start causes to create a message for the handler. all messages are queued in the looper queue and are served one at a time.

you should use normal service and do not use intentservice because you want to download simultaneously not one at a time. extend service class and in onCreate method you can create multiple threads and each thread can take a messages from onStartCommand. i do not want to copy and paste the doc example because i think it is better to read all of the doc again. if you read it you can completely understand how to create service that handles multiple task simultaneously although it has created just one thread in the example.

http://developer.android.com/guide/components/services.html

what I want to is to create a background service watching the totalDownloadQueue

i think you do not need that. just when you create downloadtask call service, your message is delivered to service class and in that class you can create blockingqueue to handle your messages by threads.

Do the threads exist after the user exit the application?

yes and maybe no. it depends on the process, if the process exists yes but if the process has been destroyed no. again read lifecycle of process to understand what process is killed or kept by android.

http://developer.android.com/guide/components/processes-and-threads.html

if this approach still can't resolve the issue about downloading files asynchronously? What other strategy should I adopt? Please provide some example code or reference so that I can modify on it.

you can use downloadmanager but it downloads sequentially.

http://developer.android.com/reference/android/app/DownloadManager.html

http://blog.vogella.com/2011/06/14/android-downloadmanager-example/