Android通過startService實現(xiàn)文件批量下載
關(guān)于startService的基本使用概述及其生命周期可參見《Android中startService基本使用方法概述》。
本文通過批量下載文件的簡單示例,演示startService以及stopService(startId)的使用流程,具體內(nèi)容如下
系統(tǒng)界面如下:

界面很簡單,就一個按鈕“批量下載文章”,通過該Activity上的按鈕啟動DownloadService。
DownloadService是用來進行下載CSDN上博客文章的服務(wù),代碼如下:
package com.ispring.startservicedemo;
import android.app.Service;
import android.content.Intent;
import android.os.Handler;
import android.os.IBinder;
import android.os.Message;
import android.util.Log;
import android.widget.Toast;
import java.io.IOException;
import java.io.InputStream;
import java.net.HttpURLConnection;
import java.net.MalformedURLException;
import java.net.URL;
import java.util.ArrayList;
import java.util.List;
public class DownloadService extends Service {
//存儲所有的startId
private List<Integer> allStartIdList = new ArrayList<>();
//存儲已經(jīng)下載完成的startId
private List<Integer> finishedStartIdList = new ArrayList<>();
private Handler handler = new Handler(){
@Override
public void handleMessage(Message msg) {
if(msg.what == 1){
String tip = (String)msg.obj;
Toast.makeText(DownloadService.this, tip, Toast.LENGTH_LONG).show();
}
}
};
class DownloadThread extends Thread {
//對應(yīng)的intent的startId信息
private int startId = 0;
//要下載的文章名稱
private String blogName = null;
//要下載的文章地址
private String blogUrl = null;
public DownloadThread(int startId, String name, String url){
this.startId = startId;
this.blogName = name;
this.blogUrl = url;
}
@Override
public void run() {
HttpURLConnection conn = null;
InputStream is = null;
try{
//下載指定的文件
URL url = new URL(this.blogUrl);
conn = (HttpURLConnection)url.openConnection();
if(conn != null){
//我們在此處得到所下載文章的輸入流,可以將其以文件的形式寫入到存儲卡上面或
//將其讀取出文本顯示在App中
is = conn.getInputStream();
}
}catch (MalformedURLException e){
e.printStackTrace();
}catch (IOException e){
e.printStackTrace();
}finally {
if(conn != null){
conn.disconnect();
}
}
finishedStartIdList.add(startId);
if(finishedStartIdList.containsAll(allStartIdList)){
String tip = "全部下載完成, 數(shù)量" + finishedStartIdList.size();
Message msg = handler.obtainMessage(1);
msg.obj = tip;
handler.sendMessage(msg);
}
Log.i("DemoLog", "stopSelf(" + startId + ")");
stopSelf(startId);
}
}
@Override
public void onCreate() {
super.onCreate();
Log.i("DemoLog", "DownloadService -> onCreate");
}
@Override
public int onStartCommand(Intent intent, int flags, int startId) {
allStartIdList.add(startId);
String name = intent.getStringExtra("name");
String url = intent.getStringExtra("url");
Log.i("DemoLog", "DownloadService -> onStartCommand, startId: " + startId + ", name: " + name);
DownloadThread downloadThread = new DownloadThread(startId, name, url);
downloadThread.start();
return START_REDELIVER_INTENT;
}
@Override
public IBinder onBind(Intent intent) {
return null;
}
@Override
public void onDestroy() {
super.onDestroy();
Log.i("DemoLog", "DownloadService -> onDestroy");
}
}
DownloadActivity的代碼如下:
package com.ispring.startservicedemo;
import android.app.Activity;
import android.content.Intent;
import android.os.Bundle;
import android.view.View;
import android.widget.Button;
import java.util.ArrayList;
import java.util.HashMap;
import java.util.Iterator;
import java.util.List;
import java.util.Map;
public class DownloadActivity extends Activity implements Button.OnClickListener {
@Override
protected void onCreate(Bundle savedInstanceState) {
super.onCreate(savedInstanceState);
setContentView(R.layout.activity_download);
}
@Override
public void onClick(View v) {
List<String> list = new ArrayList<>();
list.add("Android中startService基本使用方法概述;//www.dbjr.com.cn/article/76470.htm");
list.add("Android登陸界面實現(xiàn)清除輸入框內(nèi)容和震動效果;//www.dbjr.com.cn/article/76328.htm");
Iterator iterator = list.iterator();
while (iterator.hasNext()){
String str = (String)iterator.next();
String[] splits = str.split(";");
String name = splits[0];
String url = splits[1];
Intent intent = new Intent(this, DownloadService.class);
intent.putExtra("name", name);
intent.putExtra("url", url);
startService(intent);
}
}
}
當(dāng)我們單擊了按鈕“批量下載文章”時,我們會多次調(diào)用Activity的startService方法,其中我們在其參數(shù)intent中存儲了文章名name以及文章的地址url,由于我們多次調(diào)用了startService方法,所以會批量下載文章。
點擊按鈕后,控制臺運行結(jié)果如下所示:

調(diào)用了startService之后,Android Framework接收到了intent信息,第一次會先創(chuàng)建DownloadService的實例,然后執(zhí)行其onCreate回調(diào)方法,onCreate在Service的生命周期中只會調(diào)用一次。
調(diào)用了onCreate方法后,Android會自動回調(diào)其onStartCommand方法,其實每次調(diào)用Context的startService都會觸發(fā)onStartCommand回調(diào)方法,所以onStartCommand在Service的生命周期中可能會被調(diào)用多次。在onStartCommand方法中我們可以獲得intent和startId,intent即我們調(diào)用startService方法時傳入的參數(shù),startId是Android自動分配的,每次調(diào)用startService都會自動得到一個startId,一個startId就意味著一個job,也就是意味著一次下載任務(wù)。我們在DownloadService中有兩個字段allStartIdList和finishedStartIdList。allStartIdList存儲著所有的startId,我們在onStartCommand方法一開始就把我們得到的startId放入到allStartIdList中存儲,然后我們根據(jù)intent讀取到文章名name和文章地址url,并根據(jù)這些信息創(chuàng)建一個新的線程DownloadThread,該線程用于執(zhí)行實際的網(wǎng)絡(luò)請求下載工作。需要注意的是,為了代碼簡化起見我們在onStartCommand中只要得到intent就開辟一個新線程(DownloadThread),但是在實際生產(chǎn)環(huán)境中這樣的開銷比較大(線程新建、線程銷毀),應(yīng)該盡量使用線程池以節(jié)約開銷。
執(zhí)行了DownloadThread的start方法后,就會執(zhí)行DownloadThread線程的run方法,在該方法中我們會執(zhí)行網(wǎng)絡(luò)請求,獲取博客文章的輸入流,當(dāng)我們獲取到該輸入流之后,我們就認(rèn)為下載完成了,此時我們可以將其以文件的形式寫入到存儲卡上,也可以將其讀取出文本顯示在App上,此處我們沒有對輸入流做任何處理,我們就認(rèn)為下載完成了。下載完成后,我們把startId存入到finishedStartIdList中,finishedStartIdList存儲著所有已經(jīng)完成的job的startId。當(dāng)finishedStartIdList中已經(jīng)包含了allStartIdList的所有startId時,說明我們所有的下載任務(wù)完成了,我們會通過handler讓主線程顯示Toast告知用戶文章下載完成。在run方法的最后我們會執(zhí)行Service的stopSelf(startId)方法。需要注意的是我們在stopSelf方法中傳入了startId,這意味著我們不是直接停止Service的運行,我們只是停止該startId對應(yīng)的job的執(zhí)行,如果所有的startId所對應(yīng)的job都停止了,那么整個Service才會停止,當(dāng)整個Service停止時,才會觸發(fā)執(zhí)行Service的onDestroy回調(diào)方法。
本文只是通過批量下載博文這一簡單示例演示通過startService以及stopSelf(startId)使用Service基本使用流程,代碼沒有進行優(yōu)化,希望對大家學(xué)習(xí)Service有所幫助。
相關(guān)文章
如何在原有Android項目中快速集成React Native詳解
創(chuàng)建一個React Native項目并寫一個純的 React Native 應(yīng)用可以參考官方指南。下面這篇文章主要給大家介紹了關(guān)于如何在原有Android項目中快速集成React Native的相關(guān)資料,文中通過示例代碼介紹的非常詳細(xì),需要的朋友可以參考下。2017-12-12
Android實戰(zhàn)RecyclerView頭部尾部添加方法示例
本篇文章主要介紹了Android實戰(zhàn)RecyclerView頭部尾部添加方法示例,小編覺得挺不錯的,現(xiàn)在分享給大家,也給大家做個參考。一起跟隨小編過來看看吧2017-11-11
Android開發(fā)中button按鈕的使用及動態(tài)添加組件方法示例
這篇文章主要介紹了Android開發(fā)中button按鈕的使用及動態(tài)添加組件方法,涉及Android針對button按鈕的事件響應(yīng)及TextView動態(tài)添加相關(guān)操作技巧,需要的朋友可以參考下2017-11-11
Android之聯(lián)系人PinnedHeaderListView使用介紹
Android聯(lián)系人中的ListView是做得比較獨特的,這幾天,我把他提取出來了,寫成一個簡單的例子,留著備用,感興趣的朋友可以參考下哈2013-06-06
JS實現(xiàn)點擊參數(shù)面板按鈕顯示或隱藏數(shù)據(jù)
本文主要介紹JS實現(xiàn)點擊參數(shù)面板按鈕顯示或隱藏數(shù)據(jù)的方法,具有很好的參考價值。下面跟著小編一起來看下吧2017-03-03

