7

我有一个使用 Core Data 进行持久数据存储的 iOS 应用程序。我将 Dropbox 集成为用户执行持久存储文件 (appname.sqlite) 备份的一种方式。

UIButton 调用一个方法来查看 Dropbox 上是否已经存在文件:

            if([[DBSession sharedSession]isLinked])
            {
               NSString *folderName = [[self.dateFormatter stringFromDate:[NSDate date]] stringByReplacingOccurrencesOfString:@"/" withString:@"-"];
               NSString *destinationPath = [NSString stringWithFormat:@"/GradeBook Pro/Backup/%@/",folderName];
               self.metadataIndex = METADATA_REQUEST_BACKUP;
               [self.restClient loadMetadata:destinationPath];
            }

loadMetadata 委托方法使用现有文件的版本号(如果存在)启动上传。

-(void) restClient:(DBRestClient *)client loadedMetadata:(DBMetadata *)metadata
{
            SAVE_CORE_DATA;
            NSString *folderName = [[self.dateFormatter stringFromDate:[NSDate date]] stringByReplacingOccurrencesOfString:@"/" withString:@"-"];
            NSString *documentsDirectory = DOCUMENTS_DIRECTORY;
            NSString *sourcePath = [NSString stringWithFormat:@"%@/GradeBookPro.sqlite", documentsDirectory];
            NSString *destinationPath = [NSString stringWithFormat:@"/GradeBook Pro/Backup/%@/",folderName];
            [self.restClient uploadFile:@"GradeBookPro.sqlite" toPath:destinationPath withParentRev:[[metadata.contents lastObject]rev] fromPath:sourcePath];               
}

这适用于通过完美网络连接的合理的小文件或大文件,但上传过程中的任何小错误都会取消整个过程。我想切换到使用分块上传方法,但我不知道如何实际执行 .sqlite 文件的“分块”。

我似乎找不到任何使用分块上传的示例应用程序,我可以从中学习,文档只是说要分块提供文件。

所以,我的问题是:

  1. 分块上传是解决通过可能不稳定的网络连接上传大文件的用户问题的正确方法吗?

  2. 你能指点我“分块”文件的示例代码/应用程序/文档吗?我对 Dropbox SDK 很满意。

谢谢!

4

1 回答 1

15

我将自己回答这个问题,以防其他人有同样的问题。

事实证明,我让这种方式变得比需要的更困难。Dropbox SDK 处理文件分块,因此我只需要启动传输并对委托调用做出反应。使用的方法有:

要发送文件块 - 对于第一个块,使用 nil 作为uploadId,使用 0 作为偏移量:

- (void)uploadFileChunk:(NSString *)uploadId offset:(unsigned long long)offset fromPath:(NSString *)localPath;

发送最后一个块后,使用此方法提交上传:

- (void)uploadFile:(NSString *)filename toPath:(NSString *)parentFolder withParentRev:(NSString *)parentRev fromUploadId:(NSString *)uploadId;

我处理委托方法如下:

    - (void)restClient:(DBRestClient *)client uploadedFileChunk:(NSString *)uploadId newOffset:(unsigned long long)offset fromFile:(NSString *)localPath expires:(NSDate *)expiresDate
    {
        unsigned long long fileSize = [[[NSFileManager defaultManager]attributesOfItemAtPath:[FileHelper localDatabaseFilePath] error:nil]fileSize];

        if (offset >= fileSize)
        {
            //Upload complete, commit the file.
            [self.restClient uploadFile:DATABASE_FILENAME toPath:[FileHelper remoteDatabaseDirectory] withParentRev:self.databaseRemoteRevision fromUploadId:uploadId];
        }
        else
        {
            //Send the next chunk and update the progress HUD.
            self.progressHUD.progress = (float)((float)offset / (float)fileSize);
            [self.restClient uploadFileChunk:uploadId offset:offset fromPath:[FileHelper localDatabaseFilePath]];
        }
    }

由于我试图解决的主要问题是处理不良连接,因此我为失败的块上传实现了委托方法:

- (void)restClient:(DBRestClient *)client uploadFileChunkFailedWithError:(NSError *)error
{
    if (error != nil && (self.uploadErrorCount < DROPBOX_MAX_UPLOAD_FAILURES))
    {
        self.uploadErrorCount++;
        NSString* uploadId = [error.userInfo objectForKey:@"upload_id"];
        unsigned long long offset = [[error.userInfo objectForKey:@"offset"]unsignedLongLongValue];
        [self.restClient uploadFileChunk:uploadId offset:offset fromPath:[FileHelper localDatabaseFilePath]];
    }
    else
    {
      //show an error message and cancel the process
    }
}
于 2012-12-09T18:08:48.567 回答