微软的CodePush热更新非常难用大家都知道,速度跟被墙了没什么区别。

另外一方面,我们不希望把代码放到别人的服务器。自己写接口更新总归感觉安全一点。

so,就来自己搞个React-Native APP的热更新管理工具吧。暂且命名为hotdog。

/**************************************************/

首先我们要弄清react-native启动的原理,是直接调用jslocation的jsbundle文件和assets资源文件。

由此,我们可以自己通过的服务器接口去判断版本,并下载最新的然后替换相应的文件,然后从这个文件调用启动APP。这就像之前的一些H5APP一样做版本的管理。

以iOS为例,我们需要分以下几步去搭建这个自己的RN升级插件:

一、设置默认jsbundle地址(比如document文件夹):

1.首先打包的时候把jsbundle和assets放入copy bundle resource,每次启动后,检测document文件夹是否存在,不存在则拷贝到document文件夹,然后给RN框架读取启动。

我们建立如下的bundle文件管理类:

MXBundleHelper.h

  1. #import <Foundation/Foundation.h>
  2.  
  3. @interface MXBundleHelper : NSObject
  4.  
  5. +(NSURL *)getBundlePath;
  6.  
  7. @end

MXBundleHelper.m

  1. #import "MXBundleHelper.h"
  2. #import "RCTBundleURLProvider.h"
  3. @implementation MXBundleHelper
  4. +(NSURL *)getBundlePath{
  5. #ifdef DEBUG
  6. NSURL *jsCodeLocation = [[RCTBundleURLProvider sharedSettings] jsBundleURLForBundleRoot:@"index.ios" fallbackResource:nil];
  7. return jsCodeLocation;
  8. #else
  9. //需要存放和读取的document路径
  10. //jsbundle地址
  11. NSString *jsCachePath = [NSString stringWithFormat:@"%@/\%@",NSSearchPathForDirectoriesInDomains(NSDocumentDirectory, NSUserDomainMask, YES)[],@"main.jsbundle"];
  12. //assets文件夹地址
  13. NSString *assetsCachePath = [NSString stringWithFormat:@"%@/\%@",NSSearchPathForDirectoriesInDomains(NSDocumentDirectory, NSUserDomainMask, YES)[],@"assets"];
  14.  
  15. //判断JSBundle是否存在
  16. BOOL jsExist = [[NSFileManager defaultManager] fileExistsAtPath:jsCachePath];
  17. //如果已存在
  18. if(jsExist){
  19. NSLog(@"js已存在: %@",jsCachePath);
  20. //如果不存在
  21. }else{
  22. NSString *jsBundlePath = [[NSBundle mainBundle] pathForResource:@"main" ofType:@"jsbundle"];
  23. [[NSFileManager defaultManager] copyItemAtPath:jsBundlePath toPath:jsCachePath error:nil];
  24. NSLog(@"js已拷贝至Document: %@",jsCachePath);
  25. }
  26.  
  27. //判断assets是否存在
  28. BOOL assetsExist = [[NSFileManager defaultManager] fileExistsAtPath:assetsCachePath];
  29. //如果已存在
  30. if(assetsExist){
  31. NSLog(@"assets已存在: %@",assetsCachePath);
  32. //如果不存在
  33. }else{
  34. NSString *assetsBundlePath = [[NSBundle mainBundle] pathForResource:@"assets" ofType:nil];
  35. [[NSFileManager defaultManager] copyItemAtPath:assetsBundlePath toPath:assetsCachePath error:nil];
  36. NSLog(@"assets已拷贝至Document: %@",assetsCachePath);
  37. }
  38. return [NSURL URLWithString:jsCachePath];
  39. #endif
  40. }

二、做升级检测,有更新则下载,然后对本地文件进行替换:

假如我们不立即做更新,可以更新后替换,然后不会影响本次APP的使用,下次使用就会默认是最新的了。

如果立即更新的话,需要使用到RCTBridge类里的reload函数进行重启。

这里通过NSURLSession进行下载,然后zip解压缩等方法来实现文件的替换。

MXUpdateHelper.h

  1. #import <Foundation/Foundation.h>
  2. typedef void(^FinishBlock) (NSInteger status,id data);
  3.  
  4. @interface MXUpdateHelper : NSObject
  5. +(void)checkUpdate:(FinishBlock)finish;
  6. @end

MXUpdateHelper.m

  1. #import "MXUpdateHelper.h"
  2.  
  3. @implementation MXUpdateHelper
  4. +(void)checkUpdate:(FinishBlock)finish{
  5. NSString *url = @"http://www.xxx.com/xxxxxxx";
  6. NSMutableURLRequest *newRequest = [[NSMutableURLRequest alloc] initWithURL:[NSURL URLWithString:url]];
  7. [newRequest setHTTPMethod:@"GET"];
  8. [NSURLConnection sendAsynchronousRequest:newRequest queue:[NSOperationQueue mainQueue] completionHandler:^(NSURLResponse * response, NSData * data, NSError * connectionError) {
  9. if(connectionError == nil){
  10. //请求自己服务器的API,判断当前的JS版本是否最新
  11. /*
  12. {
  13. "version":"1.0.5",
  14. "fileUrl":"http://www.xxxx.com/xxx.zip",
  15. "message":"有新版本,请更新到我们最新的版本",
  16. "forceUpdate:"NO"
  17. }
  18. */
  19. //假如需要更新
  20. NSString *curVersion = @"1.0.0";
  21. NSString *newVersion = @"2.0.0";
  22. //一般情况下不一样,就是旧版本了
  23. if(![curVersion isEqualToString:newVersion]){
  24. finish(,data);
  25. }else{
  26. finish(,nil);
  27. }
  28. }
  29. }];
  30. }
  31. @end

三、APPdelegate中的定制,弹框,直接强制更新等

如果需要强制刷新reload,我们新建RCTView的方式也需要稍微改下,通过新建一个RCTBridge的对象。

因为RCTBridge中有reload的接口可以使用。

  1. #import "AppDelegate.h"
  2. #import "RCTBundleURLProvider.h"
  3. #import "RCTRootView.h"
  4. #import "MXBundleHelper.h"
  5. #import "MXUpdateHelper.h"
  6. #import "MXFileHelper.h"
  7. #import "SSZipArchive.h"
  8. @interface AppDelegate()<UIAlertViewDelegate>
  9. @property (nonatomic,strong) RCTBridge *bridge;
  10. @property (nonatomic,strong) NSDictionary *versionDic;
  11. @end
  12.  
  13. @implementation AppDelegate
  14.  
  15. - (BOOL)application:(UIApplication *)application didFinishLaunchingWithOptions:(NSDictionary *)launchOptions
  16. {
  17.  
  18. NSURL *jsCodeLocation;
  19. jsCodeLocation = [MXBundleHelper getBundlePath];
  20.  
  21. _bridge = [[RCTBridge alloc] initWithBundleURL:jsCodeLocation
  22. moduleProvider:nil
  23. launchOptions:launchOptions];
  24. RCTRootView *rootView = [[RCTRootView alloc] initWithBridge:_bridge moduleName:@"MXVersionManager" initialProperties:nil];
  25.  
  26. rootView.backgroundColor = [[UIColor alloc] initWithRed:1.0f green:1.0f blue:1.0f alpha:];
  27. self.window = [[UIWindow alloc] initWithFrame:[UIScreen mainScreen].bounds];
  28. UIViewController *rootViewController = [UIViewController new];
  29. rootViewController.view = rootView;
  30. self.window.rootViewController = rootViewController;
  31.  
  32. [self.window makeKeyAndVisible];
  33.  
  34. __weak AppDelegate *weakself = self;
  35. //更新检测
  36. [MXUpdateHelper checkUpdate:^(NSInteger status, id data) {
  37. if(status == ){
  38. weakself.versionDic = data;
  39. /*
  40. 这里具体关乎用户体验的方式就多种多样了,比如自动立即更新,弹框立即更新,自动下载下次打开再更新等。
  41. */
  42. UIAlertView *alert = [[UIAlertView alloc] initWithTitle:@"提示" message:data[@"message"] delegate:self cancelButtonTitle:@"取消" otherButtonTitles:@"现在更新", nil];
  43. [alert show];
  44. //进行下载,并更新
  45. //下载完,覆盖JS和assets,并reload界面
  46. // [weakself.bridge reload];
  47. }
  48. }];
  49. return YES;
  50. }
  51.  
  52. - (void)alertView:(UIAlertView *)alertView clickedButtonAtIndex:(NSInteger)buttonIndex{
  53. if(buttonIndex == ){
  54. //更新
  55. [[MXFileHelper shared] downloadFileWithURLString:_versionDic[@"fileurl"] finish:^(NSInteger status, id data) {
  56. if(status == ){
  57. NSLog(@"下载完成");
  58. NSError *error;
  59. NSString *filePath = (NSString *)data;
  60. NSString *desPath = [NSString stringWithFormat:@"%@",NSSearchPathForDirectoriesInDomains(NSDocumentDirectory, NSUserDomainMask, YES)[]];
  61. [SSZipArchive unzipFileAtPath:filePath toDestination:desPath overwrite:YES password:nil error:&error];
  62. if(!error){
  63. NSLog(@"解压成功");
  64. [_bridge reload];
  65. }else{
  66. NSLog(@"解压失败");
  67. }
  68. }
  69. }];
  70. }
  71. }

流程简单,通过接口请求版本,然后下载到document去访问。 其中需要做版本缓存,Zip的解压缩,以及文件拷贝等。

运行iOS工程可以看到效果。 初始为1.0.0版本,然后更新后升级到1.0.1版本。

demo: https://github.com/rayshen/MXHotdog

ReactNative 告别CodePush,自建热更新版本升级环境的更多相关文章

  1. ReactNative 使用微软的CodePush进行热更新,继续填坑

    1.别被开发环境骗了 在我们开发react native的时候,一键运行工程,js改了,只要cmd+R就可以刷新了.然后会轻易以为真正app上线的时候也是一样,只要app一打开就是最新的. 其实!这是 ...

  2. 在React Native中集成热更新

    最近,在项目DYTT集成了热更新,简单来说,就是不用重新下载安装包即可达到更新应用的目的,也不算教程吧,这里记录一下. 1.热更新方案 目前网上大概有两个比较广泛的方式,分别是 react-nativ ...

  3. react-native热更新之CodePush详细介绍及使用方法

    react-native热更新之CodePush详细介绍及使用方法 2018年03月04日 17:03:21 clf_programing 阅读数:7979 标签: react native热更新co ...

  4. React Native之code-push的热更新(ios android)

    React Native之code-push的热更新(ios android) React Native支持大家用React Native技术开发APP,并打包生成一个APP.在动态更新方面React ...

  5. CodePush热更新组件详细接入教程

    CodePush热更新组件详细接入教程 什么是CodePush CodePush是一个微软开发的云服务器.通过它,开发者可以直接在用户的设备上部署手机应用更新.CodePush相当于一个中心仓库,开发 ...

  6. 深度使用react-native的热更新能力,必须知道的一个shell命令

    开篇之前,先讲一个自己开发中的一个小插曲: 今天周日,iOS版 App 周一提交,周三审核通过上架,很给力.不过,中午11:30的时候,运营就反应某个页面有一个很明显的问题,页面没法拉到底部,部分信息 ...

  7. ReactNative学习笔记(四)热更新和增量更新

    概括 关于RN的热更新,网上有很多现成方案,但是一般都依赖第三方服务,我所希望的是能够自己管控所有一切,所以只能自己折腾. 热更新的思路 热更新一般都是更新JS和图片,也就是在不重新安装apk的情况下 ...

  8. react-native热更新从零到成功中的各种坑

    https://github.com/reactnativecn/react-native-pushy/blob/master/docs/guide.md Android NDK暂时没有安装 在你的项 ...

  9. 用CodePush在React Native App中做热更新

    最近在学React Native,学到了CodePush热更新. 老师讲了两种实现的方法,现将其记录一下. 相比较原生开发,使用React Native开发App不仅能节约开发成本,还能做原生开发不能 ...

随机推荐

  1. 关于mysql字段时间类型timestamp默认值为当前时间问题

    今天把应用部署到AWS上发现后台修改内容提交后程序报错,经过排查发现是更新数据的时候,有张数据表中的一个timestamp类型的字段默认值变成了"0000-00-00 00:00:00.00 ...

  2. SQL联合查询:子表任一记录与主表联合查询

    今天有网友群里提了这样一个关于SQL联合查询的需求: 一.有热心网友的方案: 二.我的方案: select * from ( select a.*,(select top 1 Id from B as ...

  3. window7 桌面新建快捷方式方法

    点击开始按钮 所有程序 找到某个文件夹点开,找到文件夹里的快捷方式图标,右键--属性-- 复制 目标:上图蓝色内容. 回到桌面,右键--新建--快捷方式--把复制的内容粘贴到  请键入对象的位置-- ...

  4. CentOS7安装docker

    1. 查看系统版本 $ cat /etc/redhat-release   2. 安装docker $  yum install docker 3.检查安装是否成功$ docker version 若 ...

  5. 实现一个基于 SharePoint 2013 的 Timecard 应用(中)

    门户视图 随着 Timecard 列表的增多,如何查找和管理这许多的 Timecard 也就成了问题.尤其对于团队经理而言,他除了自己填写的 Timecard,还要审核团队成员的 Timecard 任 ...

  6. iOS中空字符串报错

    *参考: http://www.ithao123.cn/content-8030945.html *参考: http://www.cnblogs.com/ziyi--caolu/p/4825633.h ...

  7. 腾讯开放平台 手机QQ登录 错误码:110406 解决办法

    作者:Panda Fang 出处:http://www.cnblogs.com/lonkiss/p/4204284.html 原创文章,转载请注明作者和出处,未经允许不可用于商业营利活动 腾讯开发平台 ...

  8. VS2015调试时没有启动IIS Express Web服务器 或者停止调试时 IIS Express 跟着关闭

    解决方法: 打开 解决方案资源管理器 -> 点选 Web 项目选择 -> 属性 -> Web "服务器"  去掉勾选"将服务器设置应道所有用户" ...

  9. problems during rovio build

    1. rovio的readme中使用的是catkin build, ROS tutorial中用的是catkin_make. 关于build与make的区别: build重新编译所有文件:make默认 ...

  10. Spring事物管理

    spring事务配置的五种方式 前段时间对Spring的事务配置做了比较深入的研究,在此之间对Spring的事务配置虽说也配置过,但是一直没有一个清楚的认识.通过这次的学习发觉Spring的事务配置只 ...