iOS開發(fā)實現(xiàn)下載器的基本功能(1)
今天,做了一個下載器的Demo,即從本地配置的Apache服務(wù)器上,下載指定的文件。這次,我們下載服務(wù)器根目錄下的html.mp4文件。
按照慣例,我們先創(chuàng)建一個URL對象和請求。
NSURL *url = [NSURL URLWithString:@"http://127.0.0.1/html.mp4"];
NSURLRequest *request = [NSURLRequest requestWithURL:url];
這里有兩點需要注意,第一,這個url的字符串是全英文的,如果在字符串中出現(xiàn)了中文,我們就不能直接調(diào)用URLWithString:這個方法,而是要先將url字符串存入一個字符串對象中,再將這個字符串通過
[urlString stringByAddingPercentEscapesUsingEncoding:NSUTF8StringEncoding]
方法才可以,否則無法正常請求。
由于是下載操作,我們就需要用到NSURLConnection的代理方法來實現(xiàn),前提是先創(chuàng)建連接對象和對象的代理。
//建立連接,立即執(zhí)行
[NSURLConnection connectionWithRequest:request delegate:self];
現(xiàn)在問題來了,代理不止一個可選,<NSURLConnectionDownloadDelegate>和<NSURLConnectionDataDelegate>,初次接觸,本能地選擇了第一個代理(因為從名字來看,第一個最像)。如果您的想法和我一樣,那就錯了,第一個代理中的方法實現(xiàn)后,確實可以獲得數(shù)據(jù),但是不知道數(shù)據(jù)存在了哪里,并不是我們指定的路徑,您可以嘗試一下。
好,經(jīng)過第一次的失敗,我們選擇第二個代理,進入頭文件,我們看到了四個方法:
//獲得響應(yīng) - (void)connection:(NSURLConnection *)connection didReceiveResponse:(NSURLResponse *)response; //獲取數(shù)據(jù) - (void)connection:(NSURLConnection *)connection didReceiveData:(NSData *)data; //斷開連接 - (void)connectionDidFinishLoading:(NSURLConnection *)connection; //發(fā)生錯誤 - (void)connection:(NSURLConnection *)connection didFailWithError:(NSError *)error;
我們可以清楚地了解每個方法的作用,您有興趣可以打印下每個方法的參數(shù)瞧一瞧。
這里需要補充的是,我們加了幾個屬性
/// 文件下載流 @property (strong, nonatomic) NSOutputStream *fileStream; /// 記錄文件總長度 @property (assign, nonatomic) long long fileLength; /// 文件當(dāng)前長度 @property (assign, nonatomic) long long currentFileLength;
關(guān)于NSOutputStream,還有一個NSFileHandle可以和他進行比較,只是后者會造成文件被重復(fù)追加。因此,我們選擇前者。根據(jù)類名我們可以推斷出應(yīng)該還有一個NSInputStream,沒錯,一個下載流,一個上傳流。
第一步。在獲得響應(yīng)的方法中,我們從response參數(shù)里獲得文件的總長度,并且置當(dāng)前已經(jīng)下載的文件長度是0,開啟一個保存到指定路徑的下載流,這里我們保存到桌面。
//獲得響應(yīng) - (void)connection:(NSURLConnection *)connection didReceiveResponse:(NSURLResponse *)response { self.fileLength = response.expectedContentLength; //當(dāng)前文件長度置零 self.currentFileLength = 0; self.fileStream = [[NSOutputStream alloc] initToFileAtPath:@"/Users/xxx/Desktop/html.mp4" append:YES]; [self.fileStream open]; }
第二步。我們獲得了數(shù)據(jù),如果您在這個方法中打印數(shù)據(jù),您就會發(fā)現(xiàn)當(dāng)文件夠大(幾M就行)時,這個方法會被調(diào)用多次,也就是說,分多次獲取數(shù)據(jù)。所以我們在這個方法中拼接數(shù)據(jù),同時也要避免數(shù)據(jù)拼接后造成占用過多內(nèi)存。我們累加已下載的數(shù)據(jù)的長度,計算已下載的百分比,并寫入數(shù)據(jù)流中。在計算百分比時,記得轉(zhuǎn)換類型哦,不然結(jié)果都是0,除了最后一個是1。
//獲取數(shù)據(jù) - (void)connection:(NSURLConnection *)connection didReceiveData:(NSData *)data { // NSLog(@"did receive:%@",data); self.currentFileLength += data.length; float progressPercent = (float)self.currentFileLength / self.fileLength; NSLog(@"have downloaded: %f", progressPercent); [self.fileStream write:data.bytes maxLength:data.length]; }
最后一步。實際上是兩個方法,一個是下載完成調(diào)用,一個是下載失敗調(diào)用。有一點需要注意,無論下載成功或失敗,都需要把文件輸出流關(guān)閉。
//斷開連接 - (void)connectionDidFinishLoading:(NSURLConnection *)connection { NSLog(@"連接結(jié)束"); [self.fileStream close]; } //發(fā)生錯誤 - (void)connection:(NSURLConnection *)connection didFailWithError:(NSError *)error { NSLog(@"%@",error); [self.fileStream close]; }
這只是下載功能的基本實現(xiàn),接下來會添加下載進度條,并對下載操作進行跟多優(yōu)化(多線程,斷點續(xù)傳等),最后會對下載操作進行封裝。
以上就是本文的全部內(nèi)容,希望對大家的學(xué)習(xí)有所幫助,也希望大家多多支持腳本之家。
相關(guān)文章
iOS利用AFNetworking實現(xiàn)文件上傳的示例代碼
本篇文章主要介紹了iOS利用AFNetworking實現(xiàn)文件上傳的示例代碼,小編覺得挺不錯的,現(xiàn)在分享給大家,也給大家做個參考。一起跟隨小編過來看看吧2017-02-02UITextView實現(xiàn)只允許鏈接交互不允許選擇圖片的方法
這篇文章主要介紹了UITextView實現(xiàn)只允許鏈接交互不允許選擇圖片的方法,文中介紹的非常詳細(xì),相信對大家具有一定的參考價值,需要的朋友們下面來一起看看吧。2017-03-03IOS 播放系統(tǒng)提示音使用總結(jié)(AudioToolbox)
這篇文章主要介紹了IOS 播放系統(tǒng)提示音使用總結(jié)(AudioToolbox)的相關(guān)資料,需要的朋友可以參考下2017-05-05