概述
我們在開發(fā)中經(jīng)常需要從服務器下載文件刃永,下載的內容可能有交換的信息确虱,緩存的圖片丝里,程序更新包等。我們使用URLConnection來實現(xiàn)下載卒蘸。
先看幾行代碼:
String urlDownload = "";
urlDownload = "http://www.baidu.com/img/baidu_sylogo1.gif";
URL url = new URL(urlDownload );
// 打開連接
URLConnection con = url.openConnection();
// 輸入流
InputStream is = con.getInputStream();
如上面的代碼所示雌隅,指定一個下載的目標鏈接,我們上面指定了一個圖片地址。然后構建一個URL對象澄步,調用該對象的openConnection方法來建立一個數(shù)據(jù)通路(連接)冰蘑。代碼的最后一行使用 con.getInputStream,拿到一個輸入流對象村缸,通過這個流對象我們就可以讀取到這個文件的內容了祠肥。下面要做的,就是讀取這個流梯皿,將流寫入我們的本地文件仇箱。
不過在這之前,我們還要說下這個方法:
//獲得文件的長度
int contentLength = con.getContentLength();
System.out.println("長度 :"+contentLength);
獲得文件長度的方法
ContentLength是不很熟啊东羹。它是http協(xié)議里描述頭(head)部分的描述屬性之一剂桥。實際這里是發(fā)送了一個http請求,分析了返回(response)里數(shù)據(jù)內容属提。
我們常常會把文件下載到手機的存儲卡里权逗,所以還會用到獲得存儲卡路徑的方法:
獲得存儲卡路徑,構成 保存文件的目標路徑
String dirName = "";
dirName = Environment.getExternalStorageDirectory()+"/MyDownload/";
File f = new File(dirName);
if(!f.exists())
{
f.mkdir();
}
Environment.getExternalStorageDirectory() 方法會返回一個字符串冤议,指示了存儲卡的路徑斟薇。我們拼接字符串出一個準備存放下載文件的文件夾。并先判斷文件夾是是否存在恕酸,如果不存在堪滨,則新建一個文件夾。
做完了上面的準備后蕊温,基本就能實現(xiàn)下載了袱箱。
完整代碼
下載前的準備工作:
//要下載的文件路徑
String urlDownload = "";
//urlDownload = "http://192.168.3.39/text.txt%22;
urlDownload = "http://www.baidu.com/img/baidu_sylogo1.gif%22;
// 獲得存儲卡路徑,構成 保存文件的目標路徑
String dirName = "";
dirName = Environment.getExternalStorageDirectory()+"/MyDownload/";
File f = new File(dirName);
if(!f.exists())
{
f.mkdir();
}
下載的操作
//準備拼接新的文件名(保存在存儲卡后的文件名)
String newFilename = _urlStr.substring(_urlStr.lastIndexOf("/")+1);
newFilename = _dirName + newFilename;
File file = new File(newFilename);
//如果目標文件已經(jīng)存在义矛,則刪除发笔。產(chǎn)生覆蓋舊文件的效果
if(file.exists())
{
file.delete();
}
try {
// 構造URL
URL url = new URL(_urlStr);
// 打開連接
URLConnection con = url.openConnection();
//獲得文件的長度
int contentLength = con.getContentLength();
System.out.println("長度 :"+contentLength);
// 輸入流
InputStream is = con.getInputStream();
// 1K的數(shù)據(jù)緩沖
byte[] bs = new byte[1024];
// 讀取到的數(shù)據(jù)長度
int len;
// 輸出的文件流
OutputStream os = new FileOutputStream(newFilename);
// 開始讀取
while ((len = is.read(bs)) != -1) {
os.write(bs, 0, len);
}
// 完畢,關閉所有鏈接
os.close();
is.close();
} catch (Exception e) {
e.printStackTrace();
}
進階篇 - 增加進度條提示下載進度
我們先來看下進度條
<ProgressBar
android:id="@+id/ProgressBar01"
style="?android:attr/progressBarStyleHorizontal"
android:layout_height="wrap_content"
android:visibility="visible"
android:max="100"
android:progress="1"
android:layout_width="200dp"/>
上面展示了一個水平的進度條凉翻。max屬性指定了最大值筐咧,progress指示當前的進度位置。style指定了一個現(xiàn)實風格噪矛。
“觀察者”模式
不得不說的一個是設計模式里的“觀察者”模式。觀察者模式提供了一個“讓一個觀察者去觀察一個對象铺罢,當觀察的目標發(fā)生變化時艇挨,通知給 訂閱了觀察結果的對象 ”。這句話純屬于個人理解韭赘。
它表達了幾個對象:
1.訂閱了“觀察結果的對象”缩滨,該對象會收到“觀察者”的通知。
2.觀察者對象,一個緊盯這 目標對象 的對象脉漏。它負責將 觀察的目標 的變化 通知給 “訂閱者”
3.被觀察的目標苞冯,會發(fā)生變化的目標對象,它的變化及時的都被觀察者所知侧巨。
在我們的下載時我們的幾個對象是
1.進度條舅锄,是訂閱者,它接受觀察者對象的消息司忱,來顯示自己的進度條位置皇忿。
2.觀察者,是一個handler對象坦仍。該對象適合在線程間傳遞消息鳍烁。我們就用它傳遞消息的特點,并且該對象屬于android平臺核心框架繁扎,和主界面的消息循環(huán)有聯(lián)系幔荒。
3.被觀察的目標就是下載的過程了。這個過程中下載文件的進度梳玫。
我們分別實現(xiàn)它
private Handler myHandler = new Handler(){
public void handleMessage(android.os.Message msg) {
//獲得進度爹梁,該進度由實際的操作進行通知。這里負責對通知進行處理
int progress = msg.arg1;
//設置進度條的當前位置
_ProgressBar01.setProgress(progress);
if(progress == 100)
{
Toast.makeText(getApplicationContext(), "下載成功", 0).show();
}
};
};
我們構建一個myHandler 對象汽纠,并重載了它的handleMessage方法卫键,該方法會捕獲(收到)所有發(fā)送給myHandler 的消息。我們接收消息虱朵,并根據(jù)消息攜帶的信息arg1作為當前的進度表示莉炉。
下面我們將這個myHandler 傳遞給下載的線程
new FileDownloader(urlDownload,dirName,myHandler).start();
如上面這行代碼所示,F(xiàn)ileDownloader對象是個下載器對象碴犬,它負責下載文件絮宁,同時他和觀察者myHandler關聯(lián)。有了消息服协,它就告訴這個觀察者绍昂。觀察者收到消息,通知給訂閱了消息的對象(本文為進度條)偿荷。
下面看下如何進行下載進程的:
// 構造URL
URL url = new URL(_urlStr);
// 打開連接
URLConnection con = url.openConnection();
//獲得文件的長度
int contentLength = con.getContentLength();
System.out.println("長度 :"+contentLength);
// 輸入流
InputStream is = con.getInputStream();
int hasRead = 0;//已經(jīng)讀取了多少
int progress = 0;
// 1K的數(shù)據(jù)緩沖
byte[] bs = new byte[128];
// 讀取到的數(shù)據(jù)長度
int len;
// 輸出的文件流
OutputStream os = new FileOutputStream(newFilename);
// 開始讀取
while ((len = is.read(bs)) != -1) {
os.write(bs, 0, len);
//記錄完成了的多少
hasRead +=len;
progress = (int)((double)hasRead/(double)contentLength * 100);//完成的百分比
//發(fā)送通知
Message msg = _myHandler.obtainMessage();
msg.arg1 = progress;
msg.sendToTarget();
Thread.sleep(500);//故意延遲窘游,不然進度條跑的太快看不清楚
}
// 完畢,關閉所有鏈接
os.close();
is.close();
我們記錄我們當前從服務器讀了多少字節(jié)跳纳,又知道總共需要下載多少字節(jié)忍饰。計算后,得到一個完成了多少的百分比寺庄。將這個百分比通知給 觀察者艾蓝。