0

我需要从服务器下载 .zip 文件并在解压缩后将其用于项目。我已经成功下载了 .sqlite 文件,并且可以使用右键单击(手动)来提取它,但是当我尝试使用 minizip 和 SSZipArchive 或 ZipArchive 提取它时,它会给出错误“无法打开文件”。

如果我手动提取 .sqlite 并通过在 mac 中单击右键来压缩 .sqlite 文件,那么我的代码将完全提取它,但是每当我尝试直接提取下载的文件时,就会出现此问题。

我已应用以下名称格式来编写和提取文件。

  1. 在写入文件时,名称为 XYZ.sqlite.zip 但它也给出了错误
  2. XYZ.zip 和包含的文件名是 XYZ.sqlite。这也给出了错误
  3. 我还编写了名为 XYZ.sqlite.zip 的文件,然后用 XYZ.zip 重命名它,然后解压缩,但这也不起作用。

下面是代码。

对于下载文件:

 NSURLRequest *request = [NSURLRequest requestWithURL:[NSURL URLWithString:zipURL]];
  NSArray *paths = NSSearchPathForDirectoriesInDomains(NSDocumentDirectory, NSUserDomainMask, YES);
 NSString *path = [[paths objectAtIndex:0] stringByAppendingPathComponent:[zipURL lastPathComponent]];
operation = [[AFDownloadRequestOperation alloc] initWithRequest:request targetPath:path shouldResume:YES];
 [operation.securityPolicy setAllowInvalidCertificates:YES];
 [operation setCompletionBlockWithSuccess:^(AFHTTPRequestOperation *operation, id responseObject) {
 [self unzipDBFile:path];


} failure:^(AFHTTPRequestOperation *operation, NSError *error) {
    NSLog(@"Error: %@", error);
}];

[operation setProgressiveDownloadProgressBlock:
 ^(AFDownloadRequestOperation *operation, NSInteger bytesRead, long long totalBytesRead, long long totalBytesExpected, long long totalBytesReadForFile, long long totalBytesExpectedToReadForFile) {

     NSArray *progress = [NSArray arrayWithObjects:obj_Pro,[NSNumber numberWithFloat:totalBytesReadForFile/(float)totalBytesExpectedToReadForFile], nil];

     [self performSelectorOnMainThread:@selector(updateProgressBar:) withObject:progress waitUntilDone:NO];
 }];
[operation start];

对于提取文件:

-(void)unzipDBFile:(NSString *)filePath
{

 NSString *documentDir = [NSSearchPathForDirectoriesInDomains(NSDocumentDirectory,
                                                             NSUserDomainMask, YES) objectAtIndex:0];
   NSString *outputPath = [documentDir stringByAppendingPathComponent:@"/DBFolder"];
   BOOL isExtracted =[SSZipArchive unzipFileAtPath:outputPath toDestination:documentDir];

  // NSString *filepath = [[NSBundle mainBundle] pathForResource:@"ZipFileName" ofType:@"zip"];
 //    ZipArchive *zipArchive = [[ZipArchive alloc] init];
 //    [zipArchive UnzipOpenFile:filePath];

 //    NSArray *paths = NSSearchPathForDirectoriesInDomains(NSDocumentDirectory, NSUserDomainMask, YES);
//    NSString *path = [paths objectAtIndex:0];
//    BOOL isExtracted=[zipArchive UnzipFileTo:path overWrite:YES];
//    NSLog(@"PATH=%@",path);
   if (!isExtracted)
    {
     UIAlertView *alert=[[UIAlertView alloc] initWithTitle:@"Error in extracting" message:@"" delegate:nil cancelButtonTitle:@"OK" otherButtonTitles:nil, nil];
      [alert show];
      [alert release];
   }else{
     UIAlertView *alert=[[UIAlertView alloc] initWithTitle:@"Extract Success" message:@"" delegate:nil cancelButtonTitle:@"OK" otherButtonTitles:nil, nil];
     [alert show];
     [alert release];
   }
//    [zipArchive UnzipCloseFile];
//    [zipArchive release];
}

以下是更新进度条的代码:

- (void)updateProgressBar:(NSArray *)progress
{
 UIProgressView *pgr = (UIProgressView *)[progress objectAtIndex:0];
 [pgr setProgress:[[progress objectAtIndex:1] floatValue]];
}
4

1 回答 1

0

我检查了这个问题,发现创建 .zip 的 .net 开发人员使用了 gzip 压缩,因此 minizip 无法提取它。

为了解压它,我使用了 iOS zlib 框架和下面的代码,这解决了我的问题。在实现此方法之前,您必须在标题中添加以下两件事。

#import <zlib.h>
#define CHUNK 16384

-(BOOL)unzipDBFile:(NSString *)filePath
{
NSString *documentDir = [NSSearchPathForDirectoriesInDomains(NSDocumentDirectory,
                                                             NSUserDomainMask, YES) objectAtIndex:0];
documentDir=[documentDir stringByAppendingPathComponent:DATABASE_FILE_NAME];

    gzFile file = gzopen([filePath UTF8String], "rb");
    FILE *dest = fopen([documentDir UTF8String], "w");
    unsigned char buffer[CHUNK];
    int uncompressedLength;
    while ((uncompressedLength = gzread(file, buffer, CHUNK)) ) {
        // got data out of our file
        if(fwrite(buffer, 1, uncompressedLength, dest) != uncompressedLength || ferror(dest)) {

            return FALSE;
        }
    }
    fclose(dest);
    gzclose(file);

return TRUE;
}
于 2015-01-07T09:55:31.620 回答