[Android實例] DownloadManager的使用

我們這發佈完一個應用後,通常都會根據反饋或者是迭代對應用進行更新,那麼新版本安裝包的下載就可以用到這個功能。
從Android 2.3(API level 9)開始Android用系統服務(Service)的方式提供了Download Manager來優化處理長時間的下載操作。Download Manager處理HTTP連接並監控連接中的狀態變化以及系統重啓來確保每一個下載任務順利完成。
在大多數涉及到下載的情況中使用Download Manager都是不錯的選擇,特別是當用戶切換不同的應用以後下載需要在後臺繼續進行,以及當下載任務順利完成非常重要的情況(DownloadManager對於斷點續傳功能支持很好)。
要想使用Download Manager,使用getSystemService方法請求系統的DOWNLOAD_SERVICE服務,代碼片段如下:
String serviceString =Context.DOWNLOAD_SERVICE; 
DownloadManager downloadManager; 
downloadManager = (DownloadManager)getSystemService(serviceString);

下載文件
要請求一個下載操作,需要創建一個DownloadManager.Request對象,將要請求下載的文件的Uri傳遞給DownloadManager的enqueue方法,代碼片段如下所示:

Uri uri =Uri.parse("http://developer.android.com/shareables/icon_templates-v4.0.zip"); 
DownloadManager.Request request = newRequest(uri); 
long reference =downloadManager.enqueue(request);

在這裏返回的reference變量是系統爲當前的下載請求分配的一個唯一的ID,我們可以通過這個ID重新獲得這個下載任務,進行一些自己想要進行的操作或者查詢下載的狀態以及取消下載等等。
我們可以通過addRequestHeader方法爲DownloadManager.Request對象request添加HTTP頭,也可以通過setMimeType方法重寫從服務器返回的mimetype。
我們還可以指定在什麼連接狀態下執行下載操作。setAllowedNetworkTypes方法可以用來限定在WiFi還是手機網絡下進行下載,setAllowedOverRoaming方法可以用來阻止手機在漫遊狀態下下載。
下面的代碼片段用於指定一個較大的文件只能在WiFi下進行下載:
request.setAllowedNetworkTypes(Request.NETWORK_WIFI);
Android API level 11 介紹了getRecommendedMaxBytesOverMobile類方法(靜態方法),返回一個當前手機網絡連接下的最大建議字節數,可以來判斷下載是否應該限定在WiFi條件下。調用enqueue方法之後,只要數據連接可用並且Download Manager可用,下載就會開始。要在下載完成的時候獲得一個系統通知(notification),註冊一個廣播接受者來接收ACTION_DOWNLOAD_COMPLETE廣播,這個廣播會包含一個EXTRA_DOWNLOAD_ID信息在intent中包含了已經完成的這個下載的ID,代碼片段如下所示:
IntentFilter filter = newIntentFilter(DownloadManager.ACTION_DOWNLOAD_COMPLETE); 

BroadcastReceiver receiver = newBroadcastReceiver() { 
@Override
public void onReceive(Context context, Intent intent) { 
long reference = intent.getLongExtra(DownloadManager.EXTRA_DOWNLOAD_ID,-1); 
if (myDownloadReference == reference) { 



}; 
registerReceiver(receiver, filter);
使用Download Manager的openDownloadedFile方法可以打開一個已經下載完成的文件,返回一個ParcelFileDescriptor對象。我們可以通過Download Manager來查詢下載文件的保存地址,如果在下載時制定了路徑和文件名,我們也可以直接操作文件。我們可以爲ACTION_NOTIFICATION_CLICKED action註冊一個廣播接受者,當用戶從通知欄點擊了一個下載項目或者從Downloads app點擊可一個下載的項目的時候,系統就會發出一個點擊下載項的廣播。
代碼片段如下:
IntentFilter filter = newIntentFilter(DownloadManager.ACTION_NOTIFICATION_CLICKED); 

BroadcastReceiver receiver = newBroadcastReceiver() { 
@Override
public void onReceive(Context context, Intent intent) { 
String extraID =DownloadManager.EXTRA_NOTIFICATION_CLICK_DOWNLOAD_IDS; 
long[] references = intent.getLongArrayExtra(extraID); 
for (long reference : references) 
if (reference == myDownloadReference) { 
// Do something with downloading file. 


}; 
registerReceiver(receiver, filter);
定製Download Manager Notifications的樣式
默認情況下,通知欄中會顯示被Download Manager管理的每一個download每一個Notification會顯示當前的下載進度和文件的名字
通過Download Manager可以爲每一個download request定製Notification的樣式,包括完全隱藏Notification。下面的代碼片段顯示了通過setTitle和setDescription方法來定製顯示在文件下載Notification中顯示的文字。
request.setTitle(“Earthquakes”); 
request.setDescription(“EarthquakeXML”);
setNotificationVisibility方法可以用來控制什麼時候顯示Notification,甚至是隱藏該request的Notification。有以下幾個參數:
Request.VISIBILITY_VISIBLE:在下載進行的過程中,通知欄中會一直顯示該下載的Notification,當下載完成時,該Notification會被移除,這是默認的參數值。
Request.VISIBILITY_VISIBLE_NOTIFY_COMPLETED:在下載過程中通知欄會一直顯示該下載的Notification,在下載完成後該Notification會繼續顯示,直到用戶點擊該Notification或者消除該Notification。
Request.VISIBILITY_VISIBLE_NOTIFY_ONLY_COMPLETION:只有在下載完成後該Notification纔會被顯示。
Request.VISIBILITY_HIDDEN:不顯示該下載請求的Notification。如果要使用這個參數,需要在應用的清單文件中加上DOWNLOAD_WITHOUT_NOTIFICATION權限。
指定下載保存地址
默認情況下,所有通過Download Manager下載的文件都保存在一個共享下載緩存中,使用系統生成的文件名每一個Request對象都可以制定一個下載
保存的地址,通常情況下,所有的下載文件都應該保存在外部存儲中,所以我們需要在應用清單文件中加上WRITE_EXTERNAL_STORAGE權限:
下面的代碼片段是在外部存儲中指定一個任意的保存位置的方法:
request.setDestinationUri(Uri.fromFile(f));
f是一個File對象。
如果下載的這個文件是你的應用所專用的,你可能會希望把這個文件放在你的應用在外部存儲中的一個專有文件夾中。注意這個文件夾不提供訪問控制,所以其他的應用也可以訪問這個文件夾。在這種情況下,如果你的應用卸載了,那麼在這個文件夾也會被刪除。
下面的代碼片段是指定存儲文件的路徑是應用在外部存儲中的專用文件夾的方法:
request.setDestinationInExternalFilesDir(this, 
Environment.DIRECTORY_DOWNLOADS, “Bugdroid.png”);
如果下載的文件希望被其他的應用共享,特別是那些你下載下來希望被Media Scanner掃描到的文件(比如音樂文件),那麼你可以指定你的下載路徑在外部存儲的公共文件夾之下,下面的代碼片段是將文件存放到外部存儲中的公共音樂文件夾的方法:
request.setDestinationInExternalPublicDir(Environment.DIRECTORY_MUSIC, 
"Android_Rock.mp3");
在默認的情況下,通過Download Manager下載的文件是不能被Media Scanner掃描到的,進而這些下載的文件(音樂、視頻等)就不會在Gallery和
Music Player這樣的應用中看到。
爲了讓下載的音樂文件可以被其他應用掃描到,我們需要調用Request對象的allowScaningByMediaScanner方法。
如果我們希望下載的文件可以被系統的Downloads應用掃描到並管理,我們需要調用Request對象的setVisibleInDownloadsUi方法,傳遞參數true。
取消和刪除下載
Download Manager的remove方法可以用來取消一個準備進行的下載,中止一個正在進行的下載,或者刪除一個已經完成的下載。
remove方法接受若干個download 的ID作爲參數,你可以設置一個或者幾個你想要取消的下載的ID,如下代碼段所示:
downloadManager.remove(REFERENCE_1,REFERENCE_2, REFERENCE_3);
該方法返回成功取消的下載的個數,如果一個下載被取消了,所有相關聯的文件,部分下載的文件和完全下載的文件都會被刪除。
查詢Download Manager
你可以通過查詢Download Manager來獲得下載任務的狀態,進度,以及各種細節,通過query方法返回一個包含了下載任務細節的Cursor。query方法傳遞一個DownloadManager.Query對象作爲參數,通過DownloadManager.Query對象的setFilterById方法可以篩選我們希望查詢的下載任務的ID。也可以使用setFilterByStatus方法篩選我們希望查詢的某一種狀態的下載任務,傳遞的參數是DownloadManager.STATUS_*常量,可以指定
正在進行、暫停、失敗、完成四種狀態。
Download Manager包含了一系列COLUMN_*靜態String常量,可以用來查詢Cursor中的結果列索引。我們可以查詢到下載任務的各種細節,包括狀態,文件大小,已經下載的字節數,標題,描述,URI,本地文件名和URI,媒體類型以及Media Provider download URI。
下面的代碼段是通過註冊監聽下載完成事件的廣播接受者來查詢下載完成文件的本地文件名和URI的實現方法:
@Override
public void onReceive(Context context,Intent intent) { 
long reference = intent.getLongExtra(DownloadManager.EXTRA_DOWNLOAD_ID,-1); 
if(myDownloadReference == reference) { 

Query myDownloadQuery = new Query(); 
myDownloadQuery.setFilterById(reference); 

Cursor myDownload = downloadManager.query(myDownloadQuery); 
if (myDownload.moveToFirst()) { 
int fileNameIdx = 
myDownload.getColumnIndex(DownloadManager.COLUMN_LOCAL_FILENAME); 
int fileUriIdx = 
myDownload.getColumnIndex(DownloadManager.COLUMN_LOCAL_URI); 

String fileName = myDownload.getString(fileNameIdx); 
String fileUri = myDownload.getString(fileUriIdx); 

// TODO Do something with the file. 
Log.d(TAG, fileName + " : " + fileUri); 

myDownload.close(); 


}
對於暫停和失敗的下載,我們可以通過查詢COLUMN_REASON列查詢出原因的整數碼。
對於STATUS_PAUSED狀態的下載,可以通過DownloadManager.PAUSED_*靜態常量來翻譯出原因的整數碼,進而判斷出下載是由於等待網絡連接
還是等待WiFi連接還是準備重新下載三種原因而暫停。
對於STATUS_FAILED狀態的下載,我們可以通過DownloadManager.ERROR_*來判斷失敗的原因,可能是錯誤碼(失敗原因)包括沒有存儲設備,
存儲空間不足,重複的文件名,或者HTTP errors。
下面的代碼是如何查詢出當前所有的暫停的下載任務,提取出暫停的原因以及文件名稱,下載標題以及當前進度的實現方法:
// Obtain the Download ManagerService. 
String serviceString =Context.DOWNLOAD_SERVICE; 
DownloadManager downloadManager; 
downloadManager =(DownloadManager)getSystemService(serviceString); 

// Create a query for pauseddownloads. 
Query pausedDownloadQuery = newQuery(); 
pausedDownloadQuery.setFilterByStatus(DownloadManager.STATUS_PAUSED); 

// Query the Download Manager for pauseddownloads. 
Cursor pausedDownloads =downloadManager.query(pausedDownloadQuery); 

// Find the column indexes for the data werequire. 
int reasonIdx =pausedDownloads.getColumnIndex(DownloadManager.COLUMN_REASON); 
int titleIdx =pausedDownloads.getColumnIndex(DownloadManager.COLUMN_TITLE); 
int fileSizeIdx = 
pausedDownloads.getColumnIndex(DownloadManager.COLUMN_TOTAL_SIZE_BYTES); 
int bytesDLIdx = 
pausedDownloads.getColumnIndex(DownloadManager.COLUMN_BYTES_DOWNLOADED_SO_FAR); 

// Iterate over the result Cursor. 
while (pausedDownloads.moveToNext()) { 
//Extract the data we require from the Cursor. 
String title = pausedDownloads.getString(titleIdx); 
intfileSize = pausedDownloads.getInt(fileSizeIdx); 
intbytesDL = pausedDownloads.getInt(bytesDLIdx); 

//Translate the pause reason to friendly text. 
intreason = pausedDownloads.getInt(reasonIdx); 
String reasonString = "Unknown"; 
switch (reason) { 
case DownloadManager.PAUSED_QUEUED_FOR_WIFI : 
reasonString = "Waiting for WiFi"; break; 
case DownloadManager.PAUSED_WAITING_FOR_NETWORK : 
reasonString = "Waiting for connectivity"; break; 
case DownloadManager.PAUSED_WAITING_TO_RETRY : 
reasonString = "Waiting to retry"; break; 
default : break; 


//Construct a status summary 
StringBuilder sb = new StringBuilder(); 
sb.append(title).append("/n"); 
sb.append(reasonString).append("/n"); 
sb.append("Downloaded ").append(bytesDL).append(" /" ).append(fileSize); 

//Display the status 
Log.d("DOWNLOAD", sb.toString()); 


// Close the result Cursor. 
pausedDownloads.close(); 
發表評論
所有評論
還沒有人評論,想成為第一個評論的人麼? 請在上方評論欄輸入並且點擊發布.
相關文章