iOS解压缩zip,本地保存读取html所有包

前语

最近一个对我来说相对的难点需求:从接口下载版本不一样的zip包,保存到本地,解压缩,然后读取本地文件夹内的html,这期间遇到了很多的问题,总结如下:

模块

  • 准备工作
  • 遇到的问题
  • 根据接口返回的地址下载
  • 把zip包保存本地,解压缩保存文件夹
  • 读取本地文件夹内的html,展示webView

1.准备工作

解压缩的需要第三方的解压缩库,换过几个解压库,都是c/c++写的,如:SSZipArchive,ZipArchive等。
 *先把SSZipArchive拖进工程
 *然后加入系统库 libz.dylib
 *在pch引入:
 #ifdef __OBJC__
 #import "ZipArchive.h"
 #endif

2.遇到的问题

*解压缩的库里面是c/c++写的,所以工程需要修改配置,因为oc与c/c++ 要混编 *
如果直接拖进 SSZipArchive ,编译会报错Unknown type name 'NSString',错误提示如下图:
iOS解压缩zip,本地保存读取html所有包_第1张图片
Unknown type name 'NSString'.png
一般情况下出现“Unknown type name”是头文件互相引用出现的,这里可以排除,由于源码使用是c\c++与oc混编,因为解压缩库里面是c/c++,如图:
iOS解压缩zip,本地保存读取html所有包_第2张图片
解压缩库.jpg
尝试了几种方案:
解决方案一:选择所有.c文件,将属性的 identity and type 改为Objective-C Source
sources.jpg
解决方案二:选择所有.c文件,将.c修改为.m
解决方案三:由于修改所有文件的编译类型,所有可能会导致其他包括c、c++代码的提示错误,修改如下,但是也会报错,最好是保留下面如图的配置,按照一二种方案解决。
iOS解压缩zip,本地保存读取html所有包_第3张图片
compile sources As.png
iOS解压缩zip,本地保存读取html所有包_第4张图片
Apple LLVM 8.0 - Language -C++.png
到此,整个解压缩的库算是编译通过了,但是我这边的工程又遇到很奇葩的问题,跟环信的sdk里面的好多方法名冲突了,我也表示很无语,太辣眼睛了,如图:
iOS解压缩zip,本地保存读取html所有包_第5张图片
error.png

因为环信所有的方法都是静态库点a里面的东西,我们这种免费用人家库的人,修改不了人家静态库的东西,只能自己默默修改SSZipArchive里面的方法名,我的天呢35个方法名,加班加点的,最后终于搞定。至此我也很无语,试了好几个第三方的解压缩软件,都是c/c++写的,都是跟环信的sdk的方法名冲突,只能自己一个一个修改。

3.根据接口返回的地址下载,解压保存到文件夹

#因为我们是根据接口版本号判断给的zip是不是最新的再保存到本地,然后读取本地的解压缩之后的文件夹。所以在此之前我先写了一个本地的plist来存取zip的版本号。本地plist的写法如下:
    NSFileManager *fileManager = [NSFileManager defaultManager];
    NSString *filePatch = [[NSSearchPathForDirectoriesInDomains(NSDocumentDirectory, NSUserDomainMask, YES)objectAtIndex:0]stringByAppendingPathComponent:@"OrderHtmlList.plist"];
    BOOL Exists = [fileManager fileExistsAtPath:filePatch];
    if(!Exists){
        
        NSString *plistPath = [[[NSBundle mainBundle]resourcePath]stringByAppendingPathComponent:@"OrderHtmlList.plist"];
        NSError *error;
        BOOL success = [fileManager copyItemAtPath:plistPath toPath:filePatch error:&error];
        if(!success){
            NSAssert1(0, @"错误写入文件:'%@'.", [error localizedDescription]);
        }
     }
#再根据版本号判断,是否下载。第一次先保存一个空的版本号为空,直接走网络下载,然后保存当前下载完之后的版本号,再次运行app的时候,判断保存的版本号,跟接口给的版本号是否一致,一致的话,就不用下载,不一致,就下载新的。如下判断:
    NSMutableDictionary *usersDic = [[NSMutableDictionary alloc] initWithContentsOfFile:filePatch];
    if ([LKTools isBlankString:[usersDic objectForKey:@"htmlVersion"]]) {
        //保存新html的版本号
        NSString *path = [[NSSearchPathForDirectoriesInDomains(NSDocumentDirectory, NSUserDomainMask, YES)objectAtIndex:0]stringByAppendingPathComponent:@"OrderHtmlList.plist"];
        NSMutableDictionary *applist = [[[NSMutableDictionary alloc]initWithContentsOfFile:path]mutableCopy];
        NSString *name = [applist objectForKey:@"htmlVersion"];
        name = [dic objectForKey:@"hnum"];
        [applist setObject:name forKey:@"htmlVersion"];
        [applist writeToFile:path atomically:YES];
        NSString *htmlFilePath = [NSString stringWithFormat:@"%@",SERVER_HOST([dic objectForKey:@"hurl"])];
        //下载解压缩
        [self rquestZipArchivePath:htmlFilePath andHtmlVersion:[dic objectForKey:@"htmlVersion"]];
        
    }
    else{
        
        if ([[usersDic objectForKey:@"htmlVersion"] isEqualToString:[dic objectForKey:@"hnum"]]) {
            NSLog(@"不下载不解压");
        }
        else{
            //保存新html的版本号
            NSString *path = [[NSSearchPathForDirectoriesInDomains(NSDocumentDirectory, NSUserDomainMask, YES)objectAtIndex:0]stringByAppendingPathComponent:@"OrderHtmlList.plist"];
            NSMutableDictionary *applist = [[[NSMutableDictionary alloc]initWithContentsOfFile:path]mutableCopy];
            NSString *name = [applist objectForKey:@"htmlVersion"];
            name = [dic objectForKey:@"hnum"];
            [applist setObject:name forKey:@"htmlVersion"];
            [applist writeToFile:path atomically:YES];
            
            NSString *htmlFilePath = [NSString stringWithFormat:@"%@",SERVER_HOST([dic objectForKey:@"hurl"])];
            //下载解压缩
            [self rquestZipArchivePath:htmlFilePath andHtmlVersion:[dic objectForKey:@"htmlVersion"]];
        }
    }
#pragma mark 请求zip地址
-(void)rquestZipArchivePath:(NSString *)pathUrl andHtmlVersion:(NSString *)version{
    //远程地址
    NSURL *URL = [NSURL URLWithString:pathUrl];
    //默认配置
    NSURLSessionConfiguration *configuration = [NSURLSessionConfiguration defaultSessionConfiguration];
    AFURLSessionManager *manager = [[AFURLSessionManager alloc] initWithSessionConfiguration:configuration];
    //请求
    NSURLRequest *request = [NSURLRequest requestWithURL:URL];
    NSURLSessionDownloadTask * downloadTask= [manager downloadTaskWithRequest:request progress:nil destination:^NSURL *(NSURL *targetPath, NSURLResponse *response) {
        //- block的返回值, 要求返回一个URL, 返回的这个URL就是文件的位置的路径
        
        NSString *cachesPath = [NSSearchPathForDirectoriesInDomains(NSCachesDirectory, NSUserDomainMask, YES) lastObject];
        
        //再次之前先删除本地文件夹里面相同的文件夹
        NSFileManager *fileManager = [NSFileManager defaultManager];
        NSArray *contents = [fileManager contentsOfDirectoryAtPath:cachesPath error:NULL];
        NSEnumerator *e = [contents objectEnumerator];
        NSString *filename;
        NSString *extension = @"zip";
        while ((filename = [e nextObject])) {

            if ([[filename pathExtension] isEqualToString:extension]) {

                [fileManager removeItemAtPath:[cachesPath stringByAppendingPathComponent:filename] error:NULL];
            }
        }
        NSString *path = [cachesPath stringByAppendingPathComponent:response.suggestedFilename];
        return [NSURL fileURLWithPath:path];
    } completionHandler:^(NSURLResponse *response, NSURL *filePath, NSError *error) {
        //设置下载完成操作
        // filePath就是你下载文件的位置,你可以解压,也可以直接拿来使用
        NSString *htmlFilePath = [filePath path];// 将NSURL转成NSString
        NSArray *documentArray =  NSSearchPathForDirectoriesInDomains(NSLibraryDirectory, NSUserDomainMask, YES);
        NSString *path = [[documentArray lastObject] stringByAppendingPathComponent:@"Preferences"];
        NSFileManager *fileManager = [NSFileManager defaultManager];
        [fileManager removeItemAtPath:[NSString stringWithFormat:@"%@/html",path] error:nil];
        [self releaseZipFilesWithUnzipFileAtPath:htmlFilePath Destination:path];
    }];
    [downloadTask resume];
}
#pragma mark 解压
- (void)releaseZipFilesWithUnzipFileAtPath:(NSString *)zipPath Destination:(NSString *)unzipPath{
//    NSLog(@"%@,%@",zipPath,unzipPath);
    NSError *error;

    if ([SSZipArchive unzipFileAtPath:zipPath toDestination:unzipPath overwrite:YES password:nil error:&error delegate:self]) {
        NSLog(@"success");
    }
    else{
        NSLog(@"%@",error);
    }
    
}
#pragma mark - SSZipArchiveDelegate
- (void)zipArchiveWillUnzipArchiveAtPath:(NSString *)path zipInfo:(unz_global_info)zipInfo {
    NSLog(@"将要解压。");
}
- (void)zipArchiveDidUnzipArchiveAtPath:(NSString *)path zipInfo:(unz_global_info)zipInfo unzippedPath:(NSString *)unzippedPat uniqueId:(NSString *)uniqueId {
    NSLog(@"解压完成!");
}

4.读取本地保存的html

先看一下保存解压缩之后的html的文件夹,绿色箭头指示的是我解压完之后的整个文件夹,红色的是我要进去读取的html所在的文件夹,如图:
iOS解压缩zip,本地保存读取html所有包_第6张图片
23F2CBE4-186A-473A-8E34-CEE674CD823D.png
#读取方法如下:
    NSArray *documentArray =  NSSearchPathForDirectoriesInDomains(NSLibraryDirectory, NSUserDomainMask, YES);
    NSString *path = [[documentArray lastObject] stringByAppendingPathComponent:@"Preferences"];
    NSURL *url=[NSURL fileURLWithPath:[NSString stringWithFormat:@"%@/html/%@",path,self.orderHtmlStr]];
    NSString *urlStr = [url absoluteString];
    urlStr = [urlStr stringByReplacingOccurrencesOfString:@"file://" withString:@""];
    NSURL * URL = [NSURL URLWithString:urlStr];
    NSURLRequest *request=[NSURLRequest requestWithURL:URL];
    [self.orderWebView loadRequest:request];

最后小结还有一个问题:H5的同事说他做了html的适配,但是我在我oc的webview上展示的时候还是没有适配,然后我就找原因,最后给webview加了个属性,就ok了,暂时性的解决,不知道是不是跟他前段H5没写好有没有关系,方法如下:self.orderWebView.scalesPageToFit = YES;//可缩放

你可能感兴趣的:(iOS解压缩zip,本地保存读取html所有包)