m

目前能够实现热更新的方法,总结起来有以下三种

1. 使用FaceBook 的开源框架 reactive native,使用js写原生的ios应用

ios app可以在运行时从服务器拉取最新的js文件到本地,然后执行,因为js是一门动态的

脚本语言,所以可以在运行时直接读取js文件执行,也因此能够实现ios的热更新


2. 使用lua 脚本。lua脚本如同js 一样,也能在动态时被。之前愤怒的小鸟使用

lua脚本做的一个插件 wax,可以实现使用lua写ios应用。热更新时,从服务器拉去lua脚本

然后动态的执行就可以了。遗憾的是 wax目前已经不更新了。


上面是网上现在能够搜到的热更新方法。

xcode 6 之后,苹果开放了 ios 的动态库编译权限。所谓的动态库,其实就是可以在运行时加载。

正好利用这一个特性,用来做ios的热更新。

1.

建立一个动态库,如图:

ionic ios热更新无效 ios热更新方案_bundle

动态库包含需要使用的viewCOntroller,当然可以包含任何需要使用的自定义ui和逻辑。

动态库的入口是一个jkDylib的类。它的.h和.m文件分别如下:

1. //
2. //  JKDylib.h
3. //  JKDylb
4. //
5. //  Created by wangdan on 15/7/5.
6. //  Copyright (c) 2015年 wangdan. All rights reserved.
7. //
8.   
9. #import <Foundation/Foundation.h>
10.   
11. @interface
12.   
13. -(void)showViewAfterVC:(id)fromVc inBundle:(NSBundle*)bundle;  
14.   
15. @end



.m文件

1. //
2. //  JKDylib.m
3. //  JKDylb
4. //
5. //  Created by wangdan on 15/7/5.
6. //  Copyright (c) 2015年 wangdan. All rights reserved.
7. //
8.   
9. #import "JKDylib.h"
10. #import "JKViewController.h"
11.   
12. @implementation
13.   
14. -(void)showViewAfterVC:(id)fromVc inBundle:(NSBundle*)bundle  
15. {  
16. if (fromVc == nil
17. return;  
18.     }  
19.       
20. JKViewController *vc = [[JKViewController alloc] init];  
21. UIViewController *preVc = (UIViewController
22.       
23. if (preVc.navigationController) {  
24. .navigationController pushViewController:vc animated:YES];  
25.     }  
26. else
27. UINavigationController *navi = [[UINavigationController alloc] init];  
28.  pushViewController:vc animated:YES];  
29.     }  
30.       
31. }  
32. @end



上述代码意图非常明显,

就是调用该动态库的时候

1. -(void)showViewAfterVC:(id)fromVc inBundle:(NSBundle*)bundle

在该函数中,创建一个viewController 然后使用mainBundler 的navigationController  push 新建的viewController,显示动态库的ui界面。

而动态库中的JKViewController 内容则可以根据需要随便定义。


2. 完成上述动态库的编译工作后,现在需要做的就是在主工程中,写一段加载该动态库的代码。

主工程目录如下:

ionic ios热更新无效 ios热更新方案_ionic ios热更新无效_02

在最重要的viewCotrooler里面,定义了加载动态库的方法:

1. //
2. //  ViewController.m
3. //  DylibTest
4. //
5. //  Created by wangdan on 15/7/5.
6. //  Copyright (c) 2015年 wangdan. All rights reserved.
7. //
8.   
9. #import "ViewController.h"
10. #import "AFNetWorking.h"
11.   
12. @interface
13.   
14. @end
15.   
16. @implementation
17.   
18. - (void)viewDidLoad {  
19. super viewDidLoad];  
20. self.view.backgroundColor = [UIColor whiteColor];  
21. self.title = @"bundle test";  
22.       
23. AFHTTPRequestOperationManager *manager = [[AFHTTPRequestOperationManager alloc] init];  
24. .responseSerializer = [AFJSONResponseSerializer serializer];  
25. NSURLRequest *request = [NSURLRequest requestWithURL:[NSURL URLWithString:@"http://www.baidu.com"]];  
26.  HTTPRequestOperationWithRequest:request success:^(AFHTTPRequestOperation *operation, id
27. @"request success");  
28.  failure:^(AFHTTPRequestOperation *operation, NSError
29. @"request failure");  
30.     }];  
31.       
32.       
33.       
34. UIButton *btn = [[UIButton alloc] initWithFrame:CGRectMake(0, 100, 100, 50)];  
35. .backgroundColor = [UIColor blueColor];  
36.       
37.  addTarget:self
38.  action:@selector(btnHandler)  
39.  forControlEvents:UIControlEventTouchUpInside];  
40.       
41. self.view addSubview:btn];  
42.       
43. NSString *document = [NSSearchPathForDirectoriesInDomains(NSDocumentDirectory, NSUserDomainMask, YES) lastObject];  
44.       
45. BOOL
46. @"hellow" writeToFile:[NSString stringWithFormat:@"%@/%@",document,@"hello.plist"] atomically:YES encoding:NSUTF8StringEncoding error:nil];  
47.       
48. // Do any additional setup after loading the view, typically from a nib.
49. }  
50.   
51.   
52. -(void)btnHandler  
53. {  
54.       
55. //AFHTTPRequestOperationManager *manager = [[AFHTTPRequestOperationManager alloc] init];
56. //manager.responseSerializer = [AFJSONResponseSerializer serializer];
57. // NSURLRequest *request = [NSURLRequest requestWithURL:[NSURL URLWithString:@"http://www.baidu.com"]];
58. // [manager HTTPRequestOperationWithRequest:request success:^(AFHTTPRequestOperation *operation, id responseObject) {
59. //    NSLog(@"request success");
60. // } failure:^(AFHTTPRequestOperation *operation, NSError *error) {
61. // NSLog(@"request failure");
62. //}];
63.       
64. NSString *documentDirectory = [NSSearchPathForDirectoriesInDomains(NSDocumentDirectory, NSUserDomainMask, YES) lastObject];  
65. NSString *bundlePath = [NSString stringWithFormat:@"%@/%@",documentDirectory,@"JKDylb.framework"];  
66.       
67. if (![[NSFileManager defaultManager] fileExistsAtPath:bundlePath]) {  
68. @"file not exist ,now  return");  
69. return;  
70.     }  
71. NSBundle *bundle = [NSBundle bundleWithPath:bundlePath];  
72.       
73. if (!bundle || ![bundle load]) {  
74. @"bundle load error");  
75.     }  
76.       
77.  principalClass];  
78. if
79. @"get bundle class fail");  
80. return;  
81.     }  
82. NSObject *bundleObj = [loadClass new];  
83.  performSelector:@selector(showViewAfterVC:inBundle:) withObject:self withObject:bundle];  
84.       
85. NSString *framePath = [[NSBundle mainBundle] privateFrameworksPath];  
86. @"framePath is %@",framePath);  
87.       
88. @"file attri \n %@",bundle.localizations);  
89.       
90. //    [bundleObj showViewAfterVC:self inBundle:bundle];
91. }  
92.   
93. - (void)didReceiveMemoryWarning {  
94. super didReceiveMemoryWarning];  
95. // Dispose of any resources that can be recreated.
96. }  
97.   
98. @end



viewController视图中有一个按钮,点击按钮后,从 document目录下面找到动态库(虽然此时document下并没有动态库),动态库的名称约定好味

JKDylib.framework

然后使用NSBundle 加载该动态库,具体见代码。

加载成功后,调用在动态库中实现的方法

1. [bundleObj performSelector:@selector(showViewAfterVC:inBundle:) withObject:self withObject:bundle];

编译该工程,然后运行到手机上,然后退出该程序


3. 打开itunes 然后将动态库同步到刚才的测试工程目录下。


4.在此打开测试工程程序,点击button,则会发现能够进入在动态库中定义的ui界面了。


上面工程的参考代码 在 


采用动态库方式实现热更新其实还是有一个问题,就是如何在主工程和动态库之间共享组建

比如网络组件以及其他等等第三方组件。

目前我没发现好方法,只能在动态库和主工程之间分别添加并且编译。