React Native与OC之间通信那些事儿

React Native用IOS自带的JavaScriptCore作为JS的解析引擎,普通的JS-OC通信就是React Native在OC定义一个模块方法,JS可以直接调用这个模块方法并还可以无缝衔接回调。

具体的接口调用实现方法如下所示:

  • 将OC注册进来的模块取出,调用模块中的对应函数,且将参数传入
var RCTVideo = require('react-native').NativeModules.RCTVideo;
RCTVideo.addVideoTitle('video title');
  • 利用回调参数得到访问OC的函数,并得到其返回值
RCTVideo.RNCallbackEvent('dsb',(error,callBackEvents)=>{
   if (error) {
       console.error(error);
   } else {
       AlertIOS.alert('返回值:'+JSON.stringify(callBackEvents));
   }
});
  • 利用回调参数得到访问OC的函数,并得到其返回值

callback函数:第一个参数是一个错误对象(没有发生错误的时候为null),而剩下的部分是函数的返回值。

RCTVideo.RNCallbackEvent('dsb',(error,callBackEvents)=>{
   if (error) {
       console.error(error);
   } else {
       AlertIOS.alert('返回值:'+JSON.stringify(callBackEvents));
   }
});
  • 如果想要OC访问JS,我们需要利用 NativeAppEventEmitter组件,利用其addListener进行注册监听
let ocFun = NativeAppEventEmitter.addListener(
  'eventName',
  (para) => AlertIOS.alert('被OC触发','字典数据:\n name:'+para.name+'\n age:'+para.age)
);

ocFun : 将绑定好的监听事件引用交给此变量保存。

addListener:

第一个参数:事件名

第二个参数:响应函数

注意:利用addListener进行监听,一定要对应有取消监听!

且通常取消监听都在componentWillUnmount函数中进行。如下:

  componentWillUnmount(){
    ocFun.remove();
  }
  • 如何用js构建native封装好的本地UI组件

简单地封装一个native封装好的本地视频组建的实现方法如下:

var { requireNativeComponent } = require('react-native');
module.exports = requireNativeComponent('RCTVideo', null);

现在这是 JavaScript中一个功能完整的 native video视图组件了,包括 pinch-zoom 和其他 native 手势支持, 但是我们还不能用 JavaScript来真正的控制它,所以接下来我们需要给组建添加属性和方法,具体示例如下:

class VideoView extends React.Component {
  constructor() {
    this.play = this.play.bind(this);
  }

  play = (event) => {
    if (this.props.play) {
      this.props.play(event.nativeEvent);
    }
  }

  render() {
    return (
      <View>
        <RCTVideo {...this.props} style = {styles.fullscreen} />
      </View>
    )
   }
  }

  VideoView.propTypes = {
    src: React.PropTypes.string,
    play:React.PropTypes.func,
  };

var RCTVideo = requireNativeComponent('RCTVideo', VideoView,   
    nativeOnly: {
      src: true,
      play: true,
    }
)

接下来看看其实现原理,理解react native与OC之间的通信我们首先需要了解模块配置表,接下来对模块配置表进行简单的介绍:

模块配置表

js如果要调用oc提供的接口方法,OC首先需要向JS传递它所有的模块信息。这里的具体实现方法是OC生成一份模块配置表传给JS,配置表里包括了所有模块和模块里方法的信,具体信息如下所示:

{
    "remoteModuleConfig": {
        "RCTVideo": {
            "methods": {
                "play": {
                    "type": "remote",
                    "methodID": 0
                }
            },
            "moduleID": 4
        },
        ...
     },
}

OC与js之间的调用流程

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

接下来看看JS调用OC模块方法的详细流程,包括callback回调,下面展示的是细化版本的调用流程图:

详细说明下这些步骤:

  • 1.JS端调用某个OC模块暴露出来的方法
  • 2.把上一步的调用分解为ModuleName,MethodName,arguments,再扔给MessageQueue处理。 在初始化时模块配置表上的每一个模块都生成了对应的remoteModule对象,对象里也生成了跟模块配置表里一一对应的方法,这些方法里可以拿到自身的模块名,方法名,并对callback进行一些处理,再移交给MessageQueue。
  • 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.调用到第6步说明的RCTModuleMethod生成的block。
  • 9.block里带着CallbackID和block传过来的参数去调JS里MessageQueue的方法invokeCallbackAndReturnFlushedQueue。
  • 10.MessageQueue通过CallbackID找到相应的JS callback方法。
  • 11.调用callback方法,并把OC带过来的参数一起传过去,完成回调。

概括一下整体流程如下 整个流程就是这样,简单概括下,差不多就是:JS函数调用转ModuleID/MethodID -> callback转CallbackID -> OC根据ID拿到方法 -> 处理参数 -> 调用OC方法 -> 回调CallbackID -> JS通过CallbackID拿到callback执行

参考文献

感谢您的阅读~~~

本文参与腾讯云自媒体分享计划,欢迎正在阅读的你也加入,一起分享。

发表于

我来说两句

0 条评论
登录 后参与评论

相关文章

来自专栏程序员宝库

10 种最常见的 Javascript 错误

英文:SKOWRONSKI 译文:elevenbeans elevenbeans.github.io/2018/02/05/top-10-javascript...

3578
来自专栏同步博客

Apache URL重写规则

  Apached的重写功能,即是mod_rewrite模块功能,它是apache的一个模块。它的功能非常强大,可以操作URL中的所有部分。

884
来自专栏Python小屋

Python模拟分析演员之间亲密程度

假设当前文件夹中有Excel文件“电影导演演员.xlsx”,其中数据格式如下图所示:

672
来自专栏FreeBuf

如何在32位系统中使用ROP+Return-to-dl来绕过ASLR+DEP

传统的利用return-to-plt+ROP来绕过ASLR + DEP的技术需要知道库中函数的偏移地址,而在没有libc库的情况下可以使用Return-to-d...

2817
来自专栏iKcamp

如何实现VM框架中的数据绑定

作者:佳杰 本文原创,转载请注明作者及出处 如何实现VM框架中的数据绑定 一:数据绑定概述 视图(view)和数据(model)之间的绑定 二:数据绑定目的...

3448
来自专栏Python攻城狮

Python模块smtplib让群发邮件变得简单1.smtplib模块介绍2.email.utils模块使用3.发送email的其它几个模块4.实例

SMTP(Simple Mail Transfer Protocol)即简单邮件传输协议,它是一组用于由源地址到目的地址传送邮件的规则,由它来控制信件的中转方式...

594
来自专栏Grace development

冷门PHP函数汇总

整理一些日常生活中基本用不到的PHP函数,也可以说在框架内基本都内置了,无需我们去自行使用的函数。量不多。后续在日常开发中如遇到更多的冷门,会更新本文章

411
来自专栏前端小叙

koa2使用注意点总结

get请求,ajax传入参数 获取的时候为ctx.request.query.参数名

692
来自专栏葡萄城控件技术团队

深入解析 DataGrid 过滤功能

过滤是DataGrid这样的表格控件的基本功能之一,也是非常重要的特性。微软提供的标准DataGrid相信大家都很熟悉了。本文要解析的不是标准DataGrid,...

1907
来自专栏信安之路

ret2resolve学习笔记

一是做个总结,二是做个备份。上篇文章感谢@大米指出的错误,格式化字符串漏洞还未销声匿迹!!!

810

扫码关注云+社区