iOS-OC-APP热更新,动态更新(仿QQ打开或关闭某个功能)

一.前言
iOS开发更新APP我觉得是比较坑的就是审核时间比较长,审核比较严,对于刚入行的小伙伴来说,雷区比较多;所以热更新是比较重要的;
大家也许会发现我们常用的QQ现在下来也就一百多兆,但是用了几个月后发现QQ在手机上占有一个多G的内存,特别是手机内存比较小的小伙伴,这是因为你在使用过程中,有一些功能是你下载下来的;


二.创建Framework
1.新建项目
新建一个Cocoa Touch Framework项目,然后在这个项目里面写你的新的功能,比如我创建了一个控制器,在控制器里面加载一张图和一个label;
<pre>- (void)uiConfig{
self.title = @"这是功能2";
UIImageView *imageView = [[UIImageView alloc]init];
imageView.frame = CGRectMake(0, 0, ScreenWidth, ScreenHeight);
NSData *data = [NSData dataWithContentsOfURL:[NSURL URLWithString:@"http://img4.duitang.com/uploads/item/201405/31/20140531174207_hH5u4.thumb.700_0.jpeg"]];
imageView.image = [UIImage imageWithData:data];
[self.view addSubview:imageView];
UILabel *label = [[UILabel alloc]init];
label.backgroundColor = [UIColor clearColor];
label.frame = CGRectMake(0, (ScreenHeight - 100)/2, ScreenWidth, 100);
label.numberOfLines = 0;
label.text = @"这是功能2这是功能2这是功能2这是功能2这是功能2这是功能2这是功能2这是功能2这是功能2这是功能2这是功能2这是功能2这是功能2这是功能2这是功能2";
[self.view addSubview:label];
}</pre>
2.添加Aggregate

在TARGETS里面新建一个Aggregate



3.添加Run Script脚本



4.脚本源码
<pre>

Sets the target folders and the final framework product.

如果工程名称和Framework的Target名称不一样的话,要自定义FMKNAME

例如: FMK_NAME = "MyFramework"

FMK_NAME=${PROJECT_NAME}

Install dir will be the final output to the framework.

The following line create it in the root folder of the current project.

INSTALL_DIR=${SRCROOT}/Products/${FMK_NAME}.framework

Working dir will be deleted after the framework creation.

WRK_DIR=build
DEVICE_DIR=${WRK_DIR}/Release-iphoneos/${FMK_NAME}.framework
SIMULATOR_DIR=${WRK_DIR}/Release-iphonesimulator/${FMK_NAME}.framework

-configuration ${CONFIGURATION}

Clean and Building both architectures.

xcodebuild -configuration "Release" -target "${FMK_NAME}" -sdk iphoneos clean build
xcodebuild -configuration "Release" -target "${FMK_NAME}" -sdk iphonesimulator clean build

Cleaning the oldest.

if [ -d "${INSTALL_DIR}" ]
then
rm -rf "${INSTALL_DIR}"
fi
mkdir -p "${INSTALL_DIR}"
cp -R "${DEVICE_DIR}/" "${INSTALL_DIR}/"

Uses the Lipo Tool to merge both binary files (i386 + armv6/armv7) into one Universal final product.

lipo -create "${DEVICE_DIR}/${FMK_NAME}" "${SIMULATOR_DIR}/${FMK_NAME}" -output "${INSTALL_DIR}/${FMK_NAME}"
rm -r "${WRK_DIR}"
open "${INSTALL_DIR}" </pre>
5.运行打包

运行工程,将生成的framework包压缩zip,然后上传服务器;
例如:http://7xqdun.com1.z0.glb.clouddn.com/FunctionZFJ1.framework.zip

三.创建项目
在项目中我们主要是下载和读取framework包;我们先要获取功能列表,在此我在本地写了一个功能列表,大家如果用得到可以将功能列表存放在服务器上;
1.创建功能列表数据
我添加了四个功能模块,存在NSUserDefaults里面;其中功能1和功能2有下载地址,其他的没有;功能1是个NSObject,功能2直接是一个控制器;
isopen:1表示打开,0表示关闭;
<pre>- (BOOL)application:(UIApplication *)application didFinishLaunchingWithOptions:(NSDictionary *)launchOptions {
// Override point for customization after application launch.
self.window = [[UIWindow alloc] initWithFrame:[[UIScreen mainScreen] bounds]];
self.window.backgroundColor = [UIColor whiteColor];
[self.window makeKeyAndVisible];
//添加假的功能列表
NSArray *functionList = [USER_DEFAULT objectForKey:@"functionList"];
if(functionList==nil || functionList.count==0){
NSArray *titleArr = @[@"功能1",@"功能2",@"功能3",@"功能4"];
NSArray *className = @[@"HotUpdateControl",@"ZFJViewController",@"",@""];
NSArray *classType = @[@"NSObject",@"UIViewController",@"",@""];
NSArray *downUrl = @[
@"http://7xqdun.com1.z0.glb.clouddn.com/HotMudel.framework.zip",
@"http://7xqdun.com1.z0.glb.clouddn.com/FunctionZFJ1.framework.zip",
@"",
@""];
NSMutableArray *functionArr = [[NSMutableArray alloc]init];
for (int i = 0; i<titleArr.count; i++) {
NSMutableDictionary *dict = [[NSMutableDictionary alloc]init];
[dict setObject:titleArr[i] forKey:@"name"];
[dict setObject:className[i] forKey:@"classname"];
[dict setObject:classType[i] forKey:@"classtype"];
[dict setObject:@(i) forKey:@"mid"];
[dict setObject:@"0" forKey:@"isopen"];//0 未开启 1开启了
[dict setObject:downUrl[i] forKey:@"downurl"];
[functionArr addObject:dict];
}
[USER_DEFAULT setObject:functionArr forKey:@"functionList"];
[USER_DEFAULT synchronize];
}
DynamicViewController *dvc = [[DynamicViewController alloc]init];
UINavigationController *nvc = [[UINavigationController alloc]initWithRootViewController:dvc];
self.window.rootViewController = nvc;
return YES;
} </pre>
2.展示功能列表
在功能列表主要用于展示所有打开过的功能,也就是isopen为1的所有功能;
a.获取本地所有打开的数据,然后在tableview上显示
<pre>- (void)getDataBase{
[self.dataArray removeAllObjects];
NSArray *functionList = [USER_DEFAULT objectForKey:@"functionList"];
for (NSDictionary *dict in functionList) {
NSInteger isopen = [dict[@"isopen"] integerValue];
if(isopen==1){
[self.dataArray addObject:dict];
}
}
[self.tableview reloadData];
}
</pre>

b.点击对于的tableviewcell 的时候跳转对应的framework读取出来的方法
注意,我将不同的framework存放在不同的文件夹下,以mid作为区分;

<pre>- (void)tableView:(UITableView *)tableView didSelectRowAtIndexPath:(NSIndexPath *)indexPath{
[tableView deselectRowAtIndexPath:indexPath animated:YES];
NSDictionary *dict = self.dataArray[indexPath.row];
//获取framework的路径名,我已mid区分
NSString destinationPath = [NSHomeDirectory() stringByAppendingString:[NSString stringWithFormat:@"/Documents/FunctionZFJ%@",dict[@"mid"]]];
NSArray
arrFramework = [self getFilenamelistOfType:@"framework" fromDirPath:destinationPath];
NSString *bundlePath = [NSString stringWithFormat:@"%@/%@",destinationPath,[arrFramework lastObject]];
if (![[NSFileManager defaultManager] fileExistsAtPath:bundlePath]) {
NSLog(@"文件不存在");
return;
}
NSBundle *bundle = [NSBundle bundleWithPath:bundlePath];
if (!bundle || ![bundle load]) {
NSLog(@"bundle加载出错");
}
NSString *className = dict[@"classname"];
NSString *classtype = dict[@"classtype"];
Class loadClass = [bundle classNamed:className];
if (!loadClass) {
NSLog(@"获取失败");
return;
}
if([classtype isEqualToString:@"NSObject"]){
NSObject *bundleObj = [loadClass new];
NSArray *arrVc = [bundleObj performSelector:@selector(getVcs)];
TabController *tvc = [[TabController alloc]initwithVcArray:arrVc];
[self.navigationController pushViewController:tvc animated:YES];
}else if([classtype isEqualToString:@"UIViewController"]){
UIViewController *uvc = (UIViewController *)[loadClass new];
[self.navigationController pushViewController:uvc animated:YES];
}
} </pre>
c.效果图

3.更多功能
在这里我们可以打开或者关闭某个功能;
a.获取所以功能,包括打开或者关闭状态的;然后在tableview上显示;
<pre>

pragma mark - 获取全部数据

  • (void)getDataBase{
    [self.dataArray removeAllObjects];
    NSArray *functionList = [USER_DEFAULT objectForKey:@"functionList"];
    NSMutableArray *openYES = [[NSMutableArray alloc]init];
    NSMutableArray *openNO = [[NSMutableArray alloc]init];
    for (NSDictionary *dict in functionList) {
    NSMutableDictionary *muDict = [[NSMutableDictionary alloc]initWithDictionary:dict];
    NSInteger isopen = [muDict[@"isopen"] integerValue];
    if(isopen==1){
    //已经打开的功能
    [openYES addObject:muDict];
    }else{
    //没有打开的功能
    [openNO addObject:muDict];
    }
    }

    [self.dataArray addObject:openNO];
    [self.dataArray addObject:openYES];

    [self.tableview reloadData];
    }
    </pre>

b.打开功能
打开某个功能就是下载对应的framework,把下载下来的zip包进行解压一下然后获取到framework,接着删除zip包,把framework放在对于的目录下;最后改变本地列表功能的状态;
<pre>#pragma mark - 开启某个功能 先下载数据
-(void)SSZipArchiveDataBaseWithDict:(NSMutableDictionary *)dict{
NSString *requestURL = dict[@"downurl"];
if(requestURL==nil || requestURL.length==0){
self.progresslabel.text = [NSString stringWithFormat:@"%@-没有下载地址,不能开启!",dict[@"name"]];
UIAlertController *alertController = [UIAlertController alertControllerWithTitle:@"提示" message:@"没有下载地址,不能开启" preferredStyle:UIAlertControllerStyleAlert];
UIAlertAction *sureBtn = [UIAlertAction actionWithTitle:@"确定" style:UIAlertActionStyleDefault handler:nil];
[alertController addAction:sureBtn];
[self presentViewController:alertController animated:YES completion:nil];
return;
}
//下载保存的路径
NSString *savedPath = [NSHomeDirectory() stringByAppendingString:[NSString stringWithFormat:@"/Documents/FunctionZFJ%@.framework.zip",dict[@"mid"]]];
AFHTTPRequestSerializer *serializer = [AFHTTPRequestSerializer serializer];
NSMutableURLRequest *request = [serializer requestWithMethod:@"POST" URLString:requestURL parameters:nil error:nil];
AFHTTPRequestOperation *operation = [[AFHTTPRequestOperation alloc]initWithRequest:request];
[operation setOutputStream:[NSOutputStream outputStreamToFileAtPath:savedPath append:NO]];
[operation setDownloadProgressBlock:^(NSUInteger bytesRead, long long totalBytesRead, long long totalBytesExpectedToRead) {
float progress = (float)totalBytesRead / totalBytesExpectedToRead;
self.progresslabel.text = [NSString stringWithFormat:@"%@下载进度:%.2f",dict[@"name"],progress];
}];
[operation setCompletionBlockWithSuccess:^(AFHTTPRequestOperation *operation, id responseObject) {
NSLog(@"下载成功");
NSString *destinationPath = [NSHomeDirectory() stringByAppendingString:[NSString stringWithFormat:@"/Documents/FunctionZFJ%@",dict[@"mid"]]];
//对下载下来的ZIP包进行解压
BOOL isScu = [SSZipArchive unzipFileAtPath:savedPath toDestination:destinationPath];
if(isScu){
NSLog(@"解压成功");
NSFileManager *fileMgr = [NSFileManager defaultManager];
BOOL bRet = [fileMgr fileExistsAtPath:savedPath];
if (bRet) {
[fileMgr removeItemAtPath:savedPath error:nil];//解压成功后删除压缩包
}
[dict setValue:@"1" forKey:@"isopen"];
[self updataBaseWithDict:dict];//解压成功后更新本地功能列表状态
}else{
NSLog(@"解压失败 --- 开启失败");
}
} failure:^(AFHTTPRequestOperation *operation, NSError *error) {
NSLog(@"下载失败 --- 开启失败");
}];
[operation start];
}
</pre>
更新本地数据
<pre>#pragma mark - 更新本地数据
-(void)updataBaseWithDict:(NSMutableDictionary *)dict{
NSInteger mid = [dict[@"mid"] integerValue];
NSMutableArray *functionList = [USER_DEFAULT objectForKey:@"functionList"];
NSMutableArray *dataArr = [[NSMutableArray alloc]initWithArray:functionList];
[dataArr replaceObjectAtIndex:mid withObject:dict];
[USER_DEFAULT setObject:dataArr forKey:@"functionList"];
BOOL isScu = [USER_DEFAULT synchronize];
if(isScu){
[self getDataBase];//重新获取数据 更新列表
if(self.refreshData){
self.refreshData();
}
}else{
NSLog(@"c操作失败");
}
} </pre>
c.关闭功能
关闭某个功能,也就是删除某个功能的framework,然后更改功能列表的状态;
<pre>#pragma mark - 关闭某个功能
-(void)delectFunctionZFJWithDict:(NSMutableDictionary *)dict{
NSFileManager *fileMgr = [NSFileManager defaultManager];
NSString *savedPath = [NSHomeDirectory() stringByAppendingString:[NSString stringWithFormat:@"/Documents/FunctionZFJ%@",dict[@"mid"]]];
BOOL bRet = [fileMgr fileExistsAtPath:savedPath];
if (bRet) {
NSError *err;
//关闭某个功能 就是删除本地的framework 然后修改本地功能状态
BOOL isScu = [fileMgr removeItemAtPath:savedPath error:&err];
if(isScu){
[dict setValue:@"0" forKey:@"isopen"];
[self updataBaseWithDict:dict];
}else{
NSLog(@"关闭失败");
}
}else{
NSLog(@"关闭失败");
}
} </pre>
d.效果图


四.源代码

在这里面有,两个framework的源代码,可项目的代码;
注意,如果有多个功能的framework,记住多个framework的命名在同一个功能里面不能重复,不然调取失败;
链接: http://download.csdn.net/detail/u014220518/9643039

五.效果图



转载请注明来源:http://blog.csdn.net/u014220518/article/details/52248803

最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 204,530评论 6 478
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 86,403评论 2 381
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 151,120评论 0 337
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 54,770评论 1 277
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 63,758评论 5 367
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 48,649评论 1 281
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 38,021评论 3 398
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 36,675评论 0 258
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 40,931评论 1 299
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 35,659评论 2 321
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 37,751评论 1 330
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 33,410评论 4 321
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 39,004评论 3 307
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 29,969评论 0 19
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 31,203评论 1 260
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 45,042评论 2 350
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 42,493评论 2 343

推荐阅读更多精彩内容

  • Android 自定义View的各种姿势1 Activity的显示之ViewRootImpl详解 Activity...
    passiontim阅读 171,432评论 25 707
  • Swift版本点击这里欢迎加入QQ群交流: 594119878最新更新日期:18-09-17 About A cu...
    ylgwhyh阅读 25,279评论 7 249
  • 我的同事我的战友,鸡年的到来,将把我这个水利局的大姐大送入退休的候车室,在这里等候退休的列车。 没有了事业的...
    含羞姑阅读 355评论 0 0
  • “娜娜?”忆言言一回头,就看见了躲在樱花树后面瑟瑟发抖的尹慕娜,“你怎么了?为什么在这里?” “我……我……我什么...
    慕櫻呐阅读 257评论 0 1
  • 小贝超喜欢小动物,小狗,小猫,小鸟,小兔子,这些毛茸茸的家伙们她都喜欢。就连小狗豆包咬了她后,还是没有任何芥蒂的跟...
    白色冰菊阅读 434评论 0 4