React Native通信机制详解

最近在捣鼓跨平台开发,最终选择使用React Native.这对于不太了解前端的原生开发者来说,坑还是挺多的.
今天我们来了解下RN的通信机制

react-native用iOS自带的JavaScriptCore作为JS的解析引擎,但并没有用到JavaScriptCore提供的一些可以让JS与OC互调的特性,而是实现了一套机制,这套机制可以通用在所有的JS引擎上,在没有JavaScriptCore的情况下,也可以用webview代替,实际上项目汇总已经有用webview作为引擎的实现,应该是用于兼容ios7以下没有JavaScriptCore的版本

普通的JS-OC通信很简单,OC向JS传信息有现成的接口,比如webview提供的stringByEvaluatingJavaScriptFromString方法可以直接在当前的context上执行一段JS脚本,并且可以获取执行后的返回值,这个返回值就相当于JS向OC传递信息.react-native也是以此为基础,通过各种手段,实现了在OC定义一个模块方法,JS可以直接调用这个模块方法并还可以无缝衔接回调

模块配置表

首先OC要 告诉JS她有什么模块,模块里面有什么方法,JS才知道有这些方法后才有可能去调用这些方法,这里的实现是OC生成一分模块配置表传给JS,配置表里包括了所有模块和模块里方法的信息,例如:


01.png

OC端和JS端分别有一个bridge,两个bridge都保存了同样一份模块配置表,JS调用OC模块方法时,通过bridge里的配置表把模块方法转为模块ID和方法ID传给OC,OC通过bridge的模块配置表找到对应的方法执行之,以上述代码为例,流程大致是这样:


02.png

在了解这个调用流程之前,我们先来看看OC的模块配置表是怎么来的,我们在新建一个OC模块时,JS和OC都不需要为新的模块去手动添加一些配置,模块配置是自动生成的,只要项目中有一个模块,就会把这个模块添加到配置表里,那这个模块配置表示怎样生成的呢?分以下两个步骤:
1,取所有模块类

每个模块类都实现了RCTBridgeModule接口,可以通过runtime接口objc_getClassList或者objc_copyClassList取出项目里所有类,然后逐个判断是否实现了RCTBridgeModule接口,就可以找到所有模块类,实现在RCTBridgeModuleClassByModule()方法里.
2,取出模块里暴露给JS的方法

一个模块里可以有很多方法,一些是可以暴露给JS直接调用的,一些是私有的不想暴露给JS的,怎样做到提取这些暴露的方法呢?我能想到的方法是对要暴露的方法名制定一些规则,比如用RCTExport作为前缀,然后用runtime方法class_getInstaceMethod取出所有方法名字,提取以RCTExport为前缀的方法,但是这样做恶心的地方就是每个方法必须加前缀,react-native用了另一个黑魔法似的方法解决这个问题:编译属性attribute;
在上述例子中我们看到模块方法里有句代码:RCT_EXPORT(),模块里的方法加上这个宏就可以实现暴露给JS,无需其他规则,那这个宏做了什么呢?来看看他的定义:

4.png

这个宏的作用是用编译属性attribute给二进制文件新建一个section,属于__DATA数据段,名字为RCTExport,并在这个段里加入当前方法名。编译器在编译时会找到attribute进行处理,为生成的可执行文件加入相应的内容。效果可以从linkmap看出来:
5.png

可以看到可执行文件数据段多了个RCTExport段,内容就是各个要暴露给JS的方法,这些内容是可以在运行时获取到的,在RCTBridge的RCTExportMethodByModuleID()方法里获取这些内容,提取每个方法的类名和方法名,就完成了提取模块里暴露给JS方法的工作.
整个模块类/方法提取实现在RCTRemoteModulesConfig()方法里

调用流程

接下来看看JS调用OC模块方法的详细流程,包括callBack回调,这时需要细化一下上述的调用流程图:


03.png

看起来有点复杂,从发起调用到执行回调总共11个步骤,下面来说明下:
1,JS端调用某个OC模块暴露出来的方法
2,把上一步的调用分解为ModuleName,MethodName,arguments,扔给MessageQueue处理.在初始化时模块配置表上的每一个模块都生成了对应的remoteModule对象,对象里也生成了跟模块配置表里一一对应的方法,这些方法里可以拿到自身的模块名,方法名,并对callBack进行一些处理,再移交给MessageQueue,具体实现在BatchedBridgeFactory.js的_createBridgeModule里.整个实现24行代码.
3,在这一步把JS的callback函数缓存在MessageQueue的一个成员变量里,用CallbackID代表callback.再通过保存在MessageQueue的模块配置表把上一步传进来的ModuleName和MethodName转为ModuleID和MethodID;
4,把上述步骤得到的ModuleID,MethodID,callbackID和其他参数argus传给OC.
5,OC接受到消息,通过模块配置表拿到对应的模块和方法
6,RCTModuleMethod对JS传过来的每一个参数进行处理
7,OC模块方法调用完毕,执行block回调
8,调用到第六步说明的RCTModuleMethod生成的block
9,block里带着CallbackID和block传过来的参数去调JS里MessageQueue的方法nvokeCallBackAndReturnFlushQueue
10,MessageQueue通过callbackID找到对应的JScallback方法
11,调用callback方法,并把OC带过来的参数一起传递过去,完成回调
整个流程概括为:
JS函数调用转ModuleID/MethodID—>callback转CallbackID—>OC根据ID拿到方法—>处理参数—>调用OC方法—>回调callbackID—>JS通过callbackID拿到callback执行

事件响应

上述第四步有一个问题:JS是怎样把数据传给OC,让OC去调用相应方法?
答案是通过返回值:JS不会主动传递数据给OC,在调用OC方法时,会在上述第四步把ModuleID,MethodID等数据加到一个队列里,等OC过来调JS的任意方法时,再把这个队列放回给OC,此时OC在执行这个队列里要调用的方法

总结

整个React Native的JS-OC通信机制大致就是这样了,关键点在于:模块化,模块配置表,传递ID,封装调用,事件响应,其设计思想和实现方法很值得学习借鉴。

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

推荐阅读更多精彩内容