React Native原理及坑点总结

一、React Native原理及流程介绍

(一)、RN前端调用native整体流程;

  js调native简单流程:

js调native流程.png



  js先根据moduleName和methodName,查询本地remoteModuleTable,remoteMethodTable,得出moduleIndex,methodIndex索引。再调用native暴露给js的nativeFlushQueueImmediate方法,执行调用native的逻辑。

详细调用流程.png


a、native端映射表数据结构
native端模块映射表数据结构.png

  native在启动时,将初始化由RCT_EXPORT_MODULE指定的模块,并加载RCT_EXPORT_METHOD指定的方法到RCTNativeModule中,上图中module1、module2等都是RCTNativeModule类。并利用JavaScriptCore提供的JSContextRef传递remoteModuleTable,remoteMethodTable给到rn前端;

b、rn前端映射表数据结构

  rn前端的remoteModuleTable数据:


remoteModuleT.png



  rn前端的remoteMethodTable数据:

remoteMethodT.png


c、native与rn前端互调的例子

  一个native调js,同时js又调用了native的例子:

js_native_js.png


(二)、native调用RN前端流程;

  与前者流程不一致的是,js和Native之间并未协商一个通用的js module名及module对应的方法Table,React Native目前在native端是直接硬编码,写死通知rn前端的模块名及方法名,这要求开发者按照统一的写法规范及接口文档去调用js。


  native调用js:

nativeToJs.png



  底层仍然以JavaScriptCore作为中介,实现native对js的调用;

Value Object::callAsFunction(JSObjectRef thisObj, int nArgs, const JSValueRef args[]) const {
  JSValueRef exn;
  JSValueRef result = JSC_JSObjectCallAsFunction(m_context, m_obj, thisObj, nArgs, args, &exn);
  if (!result) {
    throw JSException(m_context, exn, "Exception calling object as function");
  }
  return Value(m_context, result);
}

(三)通讯原理总结;

  React Native主要依赖对JavaScriptCore的深度应用,来实现js对native的调用,通过暴露指定的类和接口来避免触范apple的上架规定,避免像JSPatch一样无限制的调用native代码。rn js层实现dom tree和render tree的构建,但是绘制交由了native层处理。

  实现方案具有创新性,但还不能完全满足大型业务的开发需求,要对原有框架作些改进和修复才能在项目中落地。较适合简单、有热更新需要、界面元素不太复杂的业务。希望以RN代替native开发还为时尚早,除非有机构或个人React Native对安卓和iOS提高兼容性,和React Native底层趋于稳定。

  但是线程之间切换存在开销,频繁通讯将导致业务流程复杂,需要通过callback来实现。同时线程切换,一般会有10-30毫秒的耗时,对于即时要求严格的场景,应尽量避免频繁需要js与native间的通讯;

二、充电桩RNBridge层、及RN前端实现介绍;

  为了方便RN业务顺利开展,有必要对app原有功能进行封装,如"网络、数据库、蓝牙、动画"等,React Native原生支持不太好或不够完善的部分,进行补充或修复。

RNBridge层次结构.png



  RN前端项目结构图:
RN前端结构.png



  项目中主要是针对RN的动画实现机制有做一些修改,RN的动画是依赖于js线程和主线程间的通讯,但是js线程切换到主线程的时间不是一致的,时快时慢,这样会导致主线程更新动画的帧间隔不一致,如果js线程卡顿,那么主线程的动画也相应卡顿。


  针对以上问题,想到了用native的方式实现动画,依赖于js端传来的表达式,声明『动画类型、动画时长、时间循环次数』等,RN前端只是给native发送了动画指令及参数,动画过程完全由native操作。从而最大限度实现较好的性能。





三、安卓效果演示及简单介绍;

四、开发过程遇到的问题及坑点;

RN遇到的坑点.png



相关的经验总结备忘:

安卓经验总结:
a.用react-native run-android时,提示local.properties,的ndk目录、sdk目录未设置,打开设置;

b.不能下载https://jcenter.bintray.com/com/google/auto/value/auto-value/1.5.2/auto-value-1.5.2.jar,可以将ss代理设置成全局代理。或者不走shadowsocks客户端,直接由as配置ip、port、password、帐户。

c.默认rn demo加载的是本地assets的bundle,则输入以下命令:
1)mkdir android/app/src/main/assets
2)react-native bundle --platform android --dev false --entry-file index.js --bundle-output android/app/src/main/assets/index.android.bundle --assets-dest android/app/src/main/res

d.摇晃vivo,输入ip:port,之后,要重启npm服务,才可以在线打bundle包;

e.提示在本地ip的环境下,加载locolhost的请求,被跨域拦截,安装chrome跨域工具;

f.cd Awesome
react-native run-android
i.compile project(':android-jsc')
compile 'org.webkit:android-jsc:r174650'

g.单例莫名被析构,原因是当前rn js代码有问题,会强制析构ReactActivity及相关联的单例类。原因未知。
开发过程中常见的问题总结:
a.Super expression must either be null or a function, not undefined
component的首字母没有为大写

b.https://github.com/facebook/react-native/issues/14314
Packager can not find entry file index.ios.js in any of the roots... #14314
watchman watch-del-all
rm -rf node_modules && npm install
npm start -- --reset-cache

c.no bundle url
一般是shadowsocks开了全局代理导致。

d.Unable to resolve module `AccessibilityInfo`
重启电脑,并且npm start --reset-cache

e.Error: Cannot find module '/Users/liuyihao/beehome/node_modules/react-native/local-cli/cli.js'
    at Function.Module._resolveFilename (module.js:485:15)
    at Function.Module._load (module.js:437:25)
    at Function.Module.runMain (module.js:605:10)
    at startup (bootstrap_node.js:158:16)
    at bootstrap_node.js:575:3
npm ERR! code ELIFECYCLE
npm ERR! errno 1
提示npm ERR! code ELIFECYCLE
rm -rf node_modules && rm ./package-lock.json && npm install

f.
如果出现修改了的代码不生效,或者出现"Module `*` does not exist in the Haste module map"、<br>"Unable to resolve module `AccessibilityInfo`",可以尝试重启清除npm缓存;
Reset Metro Bundler cache: `rm -rf /tmp/metro-bundler-cache-*` or `npm start -- --reset-cache`.

h.
render方法的括号一定要在return后面,不可以换行,否则将crash.
return ()

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

推荐阅读更多精彩内容