iOS開發中檔案的上傳和下載功能的基本實現_IOS

來源:互聯網
上載者:User

檔案的上傳

說明:檔案上傳使用的時POST請求,通常把要上傳的資料儲存在請求體中。本文介紹如何不藉助第三方架構實現iOS開發中得檔案上傳。

  由於過程較為複雜,因此本文只貼出部分關鍵代碼。

主控制器的關鍵代碼:

複製代碼 代碼如下:

YYViewController.m
#import "YYViewController.h"

#define YYEncode(str) [str dataUsingEncoding:NSUTF8StringEncoding]

@interface YYViewController ()

@end


複製代碼 代碼如下:

@implementation YYViewController

- (void)viewDidLoad
{
    [super viewDidLoad];
    // Do any additional setup after loading the view, typically from a nib.
}

- (void)upload:(NSString *)name filename:(NSString *)filename mimeType:(NSString *)mimeType data:(NSData *)data parmas:(NSDictionary *)params
{
    // 檔案上傳
    NSURL *url = [NSURL URLWithString:@"http://192.168.1.200:8080/YYServer/upload"];
    NSMutableURLRequest *request = [NSMutableURLRequest requestWithURL:url];
    request.HTTPMethod = @"POST";
   
    // 佈建要求體
    NSMutableData *body = [NSMutableData data];
   
    /***************檔案參數***************/
    // 參數開始的標誌
    [body appendData:YYEncode(@"--YY\r\n")];
    // name : 指定參數名(必須跟伺服器端保持一致)
    // filename : 檔案名稱
    NSString *disposition = [NSString stringWithFormat:@"Content-Disposition: form-data; name=\"%@\"; filename=\"%@\"\r\n", name, filename];
    [body appendData:YYEncode(disposition)];
    NSString *type = [NSString stringWithFormat:@"Content-Type:
%@\r\n", mimeType];
    [body appendData:YYEncode(type)];
   
    [body appendData:YYEncode(@"\r\n")];
    [body appendData:data];
    [body appendData:YYEncode(@"\r\n")];
   
    /***************普通參數***************/
    [params enumerateKeysAndObjectsUsingBlock:^(id key, id obj, BOOL *stop) {
        // 參數開始的標誌
        [body appendData:YYEncode(@"--YY\r\n")];
        NSString *disposition = [NSString stringWithFormat:@"Content-Disposition: form-data; name=\"
%@\"\r\n", key];
        [body appendData:YYEncode(disposition)];

        [body appendData:YYEncode(@"\r\n")];
        [body appendData:YYEncode(obj)];
        [body appendData:YYEncode(@"\r\n")];
    }];
   
    /***************參數結束***************/
    // YY--\r\n
    [body appendData:YYEncode(@"--YY--\r\n")];
    request.HTTPBody = body;
   
    // 佈建要求頭
    // 請求體的長度
    [request setValue:[NSString stringWithFormat:@"%zd", body.length] forHTTPHeaderField:@"Content-Length"];
    // 聲明這個POST請求是個檔案上傳
    [request setValue:@"multipart/form-data; boundary=YY" forHTTPHeaderField:@"Content-Type"];
   
    // 發送請求
    [NSURLConnection sendAsynchronousRequest:request queue:[NSOperationQueue mainQueue] completionHandler:^(NSURLResponse *response, NSData *data, NSError *connectionError) {
        if (data) {
            NSDictionary *dict = [NSJSONSerialization JSONObjectWithData:data options:NSJSONReadingMutableLeaves error:nil];
            NSLog(@"%@", dict);
        } else {
            NSLog(@"上傳失敗");
        }
    }];
}

- (void)touchesBegan:(NSSet *)touches withEvent:(UIEvent *)event
{
    // Socket 實現斷點上傳
   
    //apache-tomcat-6.0.41/conf/web.xml 尋找 檔案的 mimeType
//    UIImage *image = [UIImage imageNamed:@"test"];
//    NSData *filedata = UIImagePNGRepresentation(image);
//    [self upload:@"file" filename:@"test.png" mimeType:@"image/png" data:filedata parmas:@{@"username" : @"123"}];
   
    // 給本地檔案發送一個請求
    NSURL *fileurl = [[NSBundle mainBundle] URLForResource:@"itcast.txt" withExtension:nil];
    NSURLRequest *request = [NSURLRequest requestWithURL:fileurl];
    NSURLResponse *repsonse = nil;
    NSData *data = [NSURLConnection sendSynchronousRequest:request returningResponse:&repsonse error:nil];
   
    // 得到mimeType
    NSLog(@"%@", repsonse.MIMEType);
    [self upload:@"file" filename:@"itcast.txt" mimeType:repsonse.MIMEType data:data parmas:@{
                                                                                              @"username" : @"999",
                                                                                              @"type" : @"XML"}];
}

@end


補充說明:

檔案上傳請求資料格式

部分檔案的MIMEType

多線程斷點下載
說明:本文介紹多線程斷點下載。項目中使用了蘋果內建的類,實現了同時開啟多條線程下載一個較大的檔案。因為實現過程較為複雜,所以下面貼出完整的代碼。

實現思路:下載開始,建立一個和要下載的檔案大小相同的檔案(如果要下載的檔案為100M,那麼就在沙箱中建立一個100M的檔案,然後計算每一段的下載量,開啟多條線程下載各段的資料,分別寫入對應的檔案部分)。

項目中用到的主要類如下:

完成的實現代碼如下:

主控制器中的代碼:

複製代碼 代碼如下:

#import "YYViewController.h"
#import "YYFileMultiDownloader.h"

@interface YYViewController ()
@property (nonatomic, strong) YYFileMultiDownloader *fileMultiDownloader;
@end


複製代碼 代碼如下:

@implementation YYViewController
-  (YYFileMultiDownloader *)fileMultiDownloader
{
    if (!_fileMultiDownloader) {
        _fileMultiDownloader = [[YYFileMultiDownloader alloc] init];
        // 需要下載的檔案遠程URL
        _fileMultiDownloader.url = @"http://192.168.1.200:8080/MJServer/resources/jre.zip";
        // 檔案儲存到什麼地方
        NSString *caches = [NSSearchPathForDirectoriesInDomains(NSCachesDirectory, NSUserDomainMask, YES) lastObject];
        NSString *filepath = [caches stringByAppendingPathComponent:@"jre.zip"];
        _fileMultiDownloader.destPath = filepath;
    }
    return _fileMultiDownloader;
}

- (void)viewDidLoad
{
    [super viewDidLoad];
   
}

- (void)touchesBegan:(NSSet *)touches withEvent:(UIEvent *)event
{
    [self.fileMultiDownloader start];
}

@end


自訂一個基類
複製代碼 代碼如下:

YYFileDownloader.h檔案
#import <Foundation/Foundation.h>

@interface YYFileDownloader : NSObject
{
    BOOL _downloading;
}
/**
 * 所需要下載檔案的遠程URL(串連伺服器的路徑)
 */
@property (nonatomic, copy) NSString *url;
/**
 * 檔案的儲存路徑(檔案下載到什麼地方)
 */
@property (nonatomic, copy) NSString *destPath;

/**
 * 是否正在下載(有沒有在下載, 只有下載器內部才知道)
 */
@property (nonatomic, readonly, getter = isDownloading) BOOL downloading;

/**
 * 用來監聽下載進度
 */
@property (nonatomic, copy) void (^progressHandler)(double progress);

/**
 * 開始(恢複)下載
 */
- (void)start;

/**
 * 暫停下載
 */
- (void)pause;
@end


YYFileDownloader.m檔案
複製代碼 代碼如下:

#import "YYFileDownloader.h"

@implementation YYFileDownloader
@end


下載器類繼承自YYFileDownloader這個類

YYFileSingDownloader.h檔案

複製代碼 代碼如下:

#import "YYFileDownloader.h"

@interface YYFileSingleDownloader : YYFileDownloader
/**
 *  開始的位置
 */
@property (nonatomic, assign) long long begin;
/**
 *  結束的位置
 */
@property (nonatomic, assign) long long end;
@end


YYFileSingDownloader.m檔案
複製代碼 代碼如下:

#import "YYFileSingleDownloader.h"
@interface YYFileSingleDownloader() <NSURLConnectionDataDelegate>
/**
 * 連線物件
 */
@property (nonatomic, strong) NSURLConnection *conn;

/**
 *  寫資料的檔案控制代碼
 */
@property (nonatomic, strong) NSFileHandle *writeHandle;
/**
 *  當前已下載資料的長度
 */
@property (nonatomic, assign) long long currentLength;
@end


複製代碼 代碼如下:

@implementation YYFileSingleDownloader

- (NSFileHandle *)writeHandle
{
    if (!_writeHandle) {
        _writeHandle = [NSFileHandle fileHandleForWritingAtPath:self.destPath];
    }
    return _writeHandle;
}

/**
 * 開始(恢複)下載
 */
- (void)start
{
    NSURL *url = [NSURL URLWithString:self.url];
    // 預設就是GET請求
    NSMutableURLRequest *request = [NSMutableURLRequest requestWithURL:url];
    // 佈建要求頭資訊
    NSString *value = [NSString stringWithFormat:@"bytes=%lld-%lld", self.begin + self.currentLength, self.end];
    [request setValue:value forHTTPHeaderField:@"Range"];
    self.conn = [NSURLConnection connectionWithRequest:request delegate:self];
   
    _downloading = YES;
}

/**
 * 暫停下載
 */
- (void)pause
{
    [self.conn cancel];
    self.conn = nil;
   
    _downloading = NO;
}


#pragma mark - NSURLConnectionDataDelegate 代理方法
/**
 *  1. 當接受到伺服器的響應(連通了伺服器)就會調用
 */
- (void)connection:(NSURLConnection *)connection didReceiveResponse:(NSURLResponse *)response
{
   
}

/**
 *  2. 當接受到伺服器的資料就會調用(可能會被調用多次, 每次調用只會傳遞部分資料)
 */
- (void)connection:(NSURLConnection *)connection didReceiveData:(NSData *)data
{
    // 移動到檔案的尾部
    [self.writeHandle seekToFileOffset:self.begin + self.currentLength];
    // 從當前移動的位置(檔案尾部)開始寫入資料
    [self.writeHandle writeData:data];
   
    // 累加長度
    self.currentLength += data.length;
   
    // 列印下載進度
    double progress = (double)self.currentLength / (self.end - self.begin);
    if (self.progressHandler) {
        self.progressHandler(progress);
    }
}

/**
 *  3. 當伺服器的資料接受完畢後就會調用
 */
- (void)connectionDidFinishLoading:(NSURLConnection *)connection
{
    // 清空屬性值
    self.currentLength = 0;
   
    // 關閉串連(不再輸入資料到檔案中)
    [self.writeHandle closeFile];
    self.writeHandle = nil;
}

/**
 *  請求錯誤(失敗)的時候調用(請求逾時\斷網\沒有網, 一般指用戶端錯誤)
 */
- (void)connection:(NSURLConnection *)connection didFailWithError:(NSError *)error
{
   
}

@end


設計多線程下載器(利用HMFileMultiDownloader能開啟多個線程同時下載一個檔案)

一個多線程下載器只下載一個檔案

YYFileMultiDownloader.h檔案

複製代碼 代碼如下:

#import "YYFileDownloader.h"

@interface YYFileMultiDownloader : YYFileDownloader
 
@end


YYFileMultiDownloader.m檔案
複製代碼 代碼如下:

#import "YYFileMultiDownloader.h"
#import "YYFileSingleDownloader.h"

#define YYMaxDownloadCount 4

@interface YYFileMultiDownloader()
@property (nonatomic, strong) NSMutableArray *singleDownloaders;
@property (nonatomic, assign) long long totalLength;
@end


複製代碼 代碼如下:

@implementation YYFileMultiDownloader

- (void)getFilesize
{
    NSMutableURLRequest *request = [NSMutableURLRequest requestWithURL:[NSURL URLWithString:self.url]];
    request.HTTPMethod = @"HEAD";
   
    NSURLResponse *response = nil;
#warning 這裡要用非同步請求
    [NSURLConnection sendSynchronousRequest:request returningResponse:&response error:nil];
    self.totalLength = response.expectedContentLength;
}

- (NSMutableArray *)singleDownloaders
{
    if (!_singleDownloaders) {
        _singleDownloaders = [NSMutableArray array];
       
        // 獲得檔案大小
        [self getFilesize];
       
        // 每條路徑的下載量
        long long size = 0;
        if (self.totalLength % YYMaxDownloadCount == 0) {
            size = self.totalLength / YYMaxDownloadCount;
        } else {
            size = self.totalLength / YYMaxDownloadCount + 1;
        }
       
        // 建立N個下載器
        for (int i = 0; i<YYMaxDownloadCount; i++) {
            YYFileSingleDownloader *singleDownloader = [[YYFileSingleDownloader alloc] init];
            singleDownloader.url = self.url;
            singleDownloader.destPath = self.destPath;
            singleDownloader.begin = i * size;
            singleDownloader.end = singleDownloader.begin + size - 1;
            singleDownloader.progressHandler = ^(double progress){
                NSLog(@"%d --- %f", i, progress);
            };
            [_singleDownloaders addObject:singleDownloader];
        }
       
        // 建立一個跟伺服器檔案等大小的臨時檔案
        [[NSFileManager defaultManager] createFileAtPath:self.destPath contents:nil attributes:nil];
       
        // 讓self.destPath檔案的長度是self.totalLengt
        NSFileHandle *handle = [NSFileHandle fileHandleForWritingAtPath:self.destPath];
        [handle truncateFileAtOffset:self.totalLength];
    }
    return _singleDownloaders;
}

/**
 * 開始(恢複)下載
 */
- (void)start
{
    [self.singleDownloaders makeObjectsPerformSelector:@selector(start)];
   
    _downloading = YES;
}

/**
 * 暫停下載
 */
- (void)pause
{
    [self.singleDownloaders makeObjectsPerformSelector:@selector(pause)];
    _downloading = NO;
}

@end


補充說明:如何獲得將要下載的檔案的大小?

相關文章

聯繫我們

該頁面正文內容均來源於網絡整理,並不代表阿里雲官方的觀點,該頁面所提到的產品和服務也與阿里云無關,如果該頁面內容對您造成了困擾,歡迎寫郵件給我們,收到郵件我們將在5個工作日內處理。

如果您發現本社區中有涉嫌抄襲的內容,歡迎發送郵件至: info-contact@alibabacloud.com 進行舉報並提供相關證據,工作人員會在 5 個工作天內聯絡您,一經查實,本站將立刻刪除涉嫌侵權內容。

A Free Trial That Lets You Build Big!

Start building with 50+ products and up to 12 months usage for Elastic Compute Service

  • Sales Support

    1 on 1 presale consultation

  • After-Sales Support

    24/7 Technical Support 6 Free Tickets per Quarter Faster Response

  • Alibaba Cloud offers highly flexible support services tailored to meet your exact needs.