前语
最近一个对我来说相对的难点需求:从接口下载版本不一样的zip包,保存到本地,解压缩,然后读取本地文件夹内的html,这期间遇到了很多的问题,总结如下:
???/h2>
- 准备工作
- 遇到的问题
- 根据接口返回的地址下载
- 把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',错误提示如下图:
一般情况下出现“Unknown type name”是头文件互相引用出现的,这里可以排除,由于源码使用是c\c++与oc混编,因为解压缩库里面是c/c++,如图:
尝试了几种方案:
解决方案一:选择所有.c文件,将属性的 identity and type 改为Objective-C Source
解决方案二:选择所有.c文件,将.c修改为.m
解决方案三:由于修改所有文件的编译类型,所有可能会导致其他包括c、c++代码的提示错误,修改如下,但是也会报错,最好是保留下面如图的配置,按照一二种方案解决。
到此,整个解压缩的库算是编译通过了,但是我这边的工程又遇到很奇葩的问题,跟环信的sdk里面的好多方法名冲突了,我也表示很无语,太辣眼睛了,如图:
解压缩的需要第三方的解压缩库,换过几个解压库,都是c/c++写的,如:SSZipArchive,ZipArchive等。
*先把SSZipArchive拖进工程
*然后加入系统库 libz.dylib
*在pch引入:
#ifdef __OBJC__
#import "ZipArchive.h"
#endif
如果直接拖进 SSZipArchive ,编译会报错Unknown type name 'NSString',错误提示如下图:
一般情况下出现“Unknown type name”是头文件互相引用出现的,这里可以排除,由于源码使用是c\c++与oc混编,因为解压缩库里面是c/c++,如图:
尝试了几种方案:
解决方案一:选择所有.c文件,将属性的 identity and type 改为Objective-C Source
解决方案二:选择所有.c文件,将.c修改为.m
解决方案三:由于修改所有文件的编译类型,所有可能会导致其他包括c、c++代码的提示错误,修改如下,但是也会报错,最好是保留下面如图的配置,按照一二种方案解决。
到此,整个解压缩的库算是编译通过了,但是我这边的工程又遇到很奇葩的问题,跟环信的sdk里面的好多方法名冲突了,我也表示很无语,太辣眼睛了,如图:
因为环信所有的方法都是静态库点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所在的文件夹,如图:
#读取方法如下:
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];