iOS端实现React Native差异化增量更新

扯会淡

RN具有的优势有很多(虽然坑更多,一代版本一筐坑),跨平台开发,一套代码Android和iOS通用,热更新,不用一直等苹果爸爸慢吞吞的审核流程,既然要做RN,那么RN的热更新部署肯定得学下,今天就总结一下一个刚学RN的小白对热更新的理解。

个人理解,RN的热更新有点类似App的版本更新,app内版本号与server端匹配,来判断是否要更新,替换加载的jsbundle文件,然后加载新的jsbundle文件来实现版本更新,那么实质上就是把app内要加载的jsbundle文件替换掉就OK了。

原理分析

自己整理的原理图.png

react-native打ios离线包

  • 打包命令说明
react-native bundle
Options:
--entry-file <path>          Path to the root JS file, either absolute or relative to JS root
(一般为index.js文件)
--platform [string]          Either "ios" or "android"
(RN入口文件的路径, 绝对路径或相对路径)
--transformer [string]       Specify a custom transformer to be used

--dev [boolean]              If false, warnings are disabled and the bundle is minified
(如果为false, 警告会不显示并且打出的包的大小会变小,默认为--dev true)
--prepack                    When passed, the output bundle will use the Prepack format.
(当通过时, 打包输出将使用Prepack格式化,默认为--prepack  false)
--bridge-config [string]     File name of a a JSON export of __fbBatchedBridgeConfig. Used by Prepack. Ex. ./bridgeconfig.json
(使用Prepack的一个json格式的文件__fbBatchedBridgeConfig 例如: ./bridgeconfig.json)
 --bundle-output <string>     File name where to store the resulting bundle, ex. /tmp/groups.bundle
(打包后的文件输出目录, 例: /tmp/groups.bundle)
--bundle-encoding [string]   Encoding the bundle should be written in (https://nodejs.org/api/buffer.html#buffer_buffer).[default: "utf8"]
(打离线包的格式 可参考链接https://nodejs.org/api/buffer.html#buffer_buffer.默认为utf-8格式)
---sourcemap-output [string]  File name where to store the sourcemap file for resulting bundle, ex. /tmp/groups.map
(生成Source Map,但0.14之后不再自动生成source map,需要手动指定这个参数。例: /tmp/groups.map)
--assets-dest [string]       Directory name where to store assets referenced in the bundle
(打包时图片资源的存储路径)
--verbose                    Enables logging
(显示打包过程)
--reset-cache                Removes cached files
(移除缓存文件)
--config [string]            Path to the CLI configuration file
(命令行的配置文件路径)
  • 具体操作
    1.cd [项目路径]
    2.在react-native根目录下的ios目录下新建bundle文件夹(mkdir ./ios/bundle)(注意:输入打包命令前必须先新建bundle文件夹)
    3.打包命令:react-native bundle --entry-file index.js --platform ios --dev false --bundle-output ./ios/bundle/index.ios.jsbundle --assets-dest ./ios/bundle/
    4.结果展示


    生成的jsbundle离线包

    一般为下面这种


    生成的jsbundle离线包

patches.pad差异化文件终端生成方案

利用google的diff文件(资料查出来,这个比较受欢迎,同时也兼容Objective-C),github地址:https://github.com/google/dif...

  • $ git clone https://github.com/LiuC520/no...
  • $ cd nodediffpatch && npm i
  • $ sudo npm link
  • 把新旧文件放入nodediffpatch/patch目录下


    新旧离线包
  • 终端输入:patbundle patch -o test01old.jsbundle -n test01new.jsbundle


    生成的差异化文件

iOS实现生成差异化文件

简单方法:把diff-match-patch实现源码拖进工程中


选择源码
拖进工程中

导入#import "DiffMatchPatch.h"开始使用,下面演示用l1.txt和l2.txt文件来展示,可以比较直观的看出效果
l1.txt文本:123
l2.txt文本:12345

- (void)demo1{
  // 获取l1.txt文件路径
  NSString *path01 = [[NSBundle mainBundle]pathForResource:@"l1" ofType:@"txt"];
  // 根据l1.txt文件路径获取data内容
  NSData *data01 = [NSData dataWithContentsOfFile:path01];
  // 将data内容转换成字符串格式
  NSString *str01 = [[NSString alloc] initWithData:data01 encoding:NSUTF8StringEncoding];
  // 获取l2.txt文件路径
  NSString *path02 = [[NSBundle mainBundle]pathForResource:@"l2" ofType:@"txt"];
  // 根据l2.txt文件路径获取data内容
  NSData *data02 = [NSData dataWithContentsOfFile:path02];
  // 将data内容转换成字符串格式
  NSString *str02 = [[NSString alloc] initWithData:data02 encoding:NSUTF8StringEncoding];
  // 创建DiffMatchPatch工具类对象
  DiffMatchPatch *patch = [[DiffMatchPatch alloc]init];
  // 对比文件内容
  // 执行该语句之后会在bundle目录下生成patches.bat文件(差异补丁文件)
  NSMutableArray *patchesArr = [patch diff_mainOfOldString:str01 andNewString:str02 checkLines:YES];
  // 生成差异补丁包
  NSArray *patchesArr1 = [patch patch_makeFromDiffs:patchesArr];
  // 解析补丁包
  NSArray *newArray = [patch patch_apply:patchesArr1 toString:str01];
  //写入到新文件(注意:这边为了在PC端更加直观的看,直接写入到绝对路径)
  BOOL isTrue = [newArray[0] writeToFile:@"/Users/devil/Desktop/自己的/RNPlatForm/ios/l1.txt" atomically:YES encoding:NSUTF8StringEncoding error:nil];
  if (isTrue) {
    NSLog(@"写入成功");
  }else{
    NSLog(@"写入失败");
  }
}

执行代码后:
l1.txt文本:12345

iOS实现patches.pat与旧jsbundle离线包合并得到新的jsbundle离线包

- (void)demo2{
  // 获取l1.txt文件路径
  NSString *path01 = [[NSBundle mainBundle]pathForResource:@"l1" ofType:@"txt"];
  // 根据l1.txt文件路径获取data内容
  NSData *data01 = [NSData dataWithContentsOfFile:path01];
  // 将data内容转换成字符串格式
  NSString *str01 = [[NSString alloc] initWithData:data01 encoding:NSUTF8StringEncoding];
  // 创建DiffMatchPatch工具类对象
  DiffMatchPatch *patch = [[DiffMatchPatch alloc]init];
  // 获取差异化文件包路径
  NSString *patchesPath = [[NSBundle mainBundle]pathForResource:@"patches.pat" ofType:nil];
  //获取差异化文件内容
  NSData *patchesData = [NSData dataWithContentsOfFile:patchesPath];
  //解析差异化文件内容
  NSString *patchesStr = [[NSString alloc]initWithData:patchesData encoding:NSUTF8StringEncoding];
  //转换pat
  NSMutableArray *patchesArr = [patch patch_fromText:patchesStr error:nil];
  // 解析补丁包
  NSArray *newArray = [patch patch_apply:patchesArr toString:str01];
  //获取新文件路径
//  NSString *newFilePath = [[NSBundle mainBundle]pathForResource:@"text3" ofType:@"txt"];
  //写入到新文件(注意:这边为了在PC端更加直观的看,直接写入到绝对路径)
  BOOL isTrue = [newArray[0] writeToFile:@"/Users/devil/Desktop/自己的/RNPlatForm/ios/text3.txt" atomically:YES encoding:NSUTF8StringEncoding error:nil];
  if (isTrue) {
    NSLog(@"写入成功");
  }else{
    NSLog(@"写入失败");
  }
}

实现本地更新离线包

//创建两个按钮,第一个按钮跳转RN界面,加载jsbundle包,第二个按钮负责更新jsbundle包
UIButton *btn4 = [[UIButton alloc]init];
  [btn4 setTitle:@"第五个" forState:UIControlStateNormal];
  [btn4 setTitleColor:[UIColor blackColor] forState:UIControlStateNormal];
  btn4.frame = CGRectMake(40, 170, 60, 30);
  [btn4 addTarget:self action:@selector(clickFifth) forControlEvents:UIControlEventTouchUpInside];
  [self.view addSubview:btn4];
  
  UIButton *btn5 = [[UIButton alloc]init];
  [btn5 setTitle:@"更新第五个界面" forState:UIControlStateNormal];
  [btn5 setTitleColor:[UIColor blackColor] forState:UIControlStateNormal];
  btn5.frame = CGRectMake(40, 200, 60, 30);
  [btn5 addTarget:self action:@selector(demo2) forControlEvents:UIControlEventTouchUpInside];
  [self.view addSubview:btn5];
//btn4按钮点击事件
- (void)clickFifth{
  NSURL *jsCodeLocation;
  jsCodeLocation = [[NSBundle mainBundle]URLForResource:@"test01old" withExtension:@"jsbundle"];
  [self creactRNPath:jsCodeLocation moduleName:@"test01platcode"]; 
}

- (void)creactRNPath:(NSURL *)jsCodeLocation moduleName:(NSString *)moduleName{
  RCTRootView *rootView = [[RCTRootView alloc] initWithBundleURL:jsCodeLocation
                                                      moduleName:moduleName
                                               initialProperties:nil
                                                   launchOptions:nil];
  rootView.backgroundColor = [[UIColor alloc] initWithRed:1.0f green:1.0f blue:1.0f alpha:1];
  UIViewController *rootViewController = [[UIViewController alloc]init];
  rootViewController.view = rootView;
  [rootViewController.navigationController setNavigationBarHidden:YES animated:YES];
  [self.navigationController pushViewController:rootViewController animated:YES];
}
//btn5按钮点击事件
- (void)demo2{
  // 获取test01old.jsbundle文件路径
  NSString *path01 = [[NSBundle mainBundle]pathForResource:@"test01old" ofType:@"jsbundle"];
  // 根据test01old.jsbundle文件路径获取data内容
  NSData *data01 = [NSData dataWithContentsOfFile:path01];
  // 将data内容转换成字符串格式
  NSString *str01 = [[NSString alloc] initWithData:data01 encoding:NSUTF8StringEncoding];
  // 创建DiffMatchPatch工具类对象
  DiffMatchPatch *patch = [[DiffMatchPatch alloc]init];
  // 获取差异化文件包路径
  NSString *patchesPath = [[NSBundle mainBundle]pathForResource:@"test01patches.pat" ofType:nil];
  //获取差异化文件内容
  NSData *patchesData = [NSData dataWithContentsOfFile:patchesPath];
  //解析差异化文件内容
  NSString *patchesStr = [[NSString alloc]initWithData:patchesData encoding:NSUTF8StringEncoding];
  //转换pat
  NSMutableArray *patchesArr = [patch patch_fromText:patchesStr error:nil];
  // 解析补丁包
  NSArray *newArray = [patch patch_apply:patchesArr toString:str01];
  //写入到新文件
  BOOL isTrue = [newArray[0] writeToFile:path01 atomically:YES encoding:NSUTF8StringEncoding error:nil];
  if (isTrue) {
    NSLog(@"写入成功");
  }else{
    NSLog(@"写入失败");
  }
}
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 213,254评论 6 492
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 90,875评论 3 387
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 158,682评论 0 348
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 56,896评论 1 285
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 66,015评论 6 385
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 50,152评论 1 291
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 39,208评论 3 412
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 37,962评论 0 268
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 44,388评论 1 304
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 36,700评论 2 327
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 38,867评论 1 341
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 34,551评论 4 335
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 40,186评论 3 317
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 30,901评论 0 21
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 32,142评论 1 267
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 46,689评论 2 362
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 43,757评论 2 351

推荐阅读更多精彩内容