【REACT NATIVE 系列教程之十二】REACT NATIVE(JS/ES)与IOS(OBJECT-C)交互通信

一用到跨平台的引擎必然要有引擎与各平台原生进行交互通信的需要。那么Himi先讲解React Native与iOS之间的通信交互。

本篇主要分为两部分讲解:(关于其中讲解的OC语法等不介绍,不懂的请自行学习)

1. React Native 访问iOS

2. iOS访问React Native

一:React Native 访问iOS

1.我们想要JS调用OC函数,就要实现一个“RCTBridgeModule”协议的Objective-C类

所以首先我们先创建一个oc新类, Himi这里起名为:TestOJO (O: object-c,J: javaScript )

2. TestOJO.h

#import<Foundation/Foundation.h>
#import"RCTBridgeModule.h"

@interfaceTestOJO:NSObject<RCTBridgeModule>

@end


引入:#import “RCTBridgeModule.h” 且使用<RCTBridgeModule> 接口,

3.为了实现RCTBridgeModule协议,类需要包含RCT_EXPORT_MODULE()宏(这个宏也可以添加一个参数用来指定在Javascript中访问这个模块的名字。如果你不指定,认就会使用这个Objective-C类的名字。)

4. 在TestOJO.m中添加如下:

RCT_EXPORT_MODULE();
//桥接到Javascript的方法返回值类型必须是void。ReactNative的桥接操作是异步的,所以要返回结果给Javascript,必须通过回调或者触发事件来进行
RCT_EXPORT_METHOD(j2oFun1:(Nsstring*)dataStringdateNumber:(int)dateNumber)
{
NSLog(@"jscalliOSfunctionj2oFun1\ndataString:%@|dateNumber:%d",dataString,dateNumber);
}

想要将oc的函数导出给js进行调用,那么就需要进行声明。声明通过RCT_EXPORT_METHOD()宏来实现:

j2oFun1:函数名,后续是两个参数,分别是Nsstring 和 int 类型数据。

调用成功后,我们输出这两个传来的值到控制台。

注意:Javascript调用的OC函数,此函数返回值类型必须是void。由于React Native的桥接操作是异步的,所以要返回结果给Javascript,必须通过回调参数进行 后续详细讲解。

从js传来的参数我们可以依靠自动类型转换的特性,跳过手动的类型转换(RCTConvert,下面详细介绍),在定义函数参数类型时,直接写上对应想要的数据类型,例如NSData等。

5. 下面看js调用代码段:

varTestOJO=require('react-native').NativeModules.TestOJO;
TestOJO.j2oFun1('Himi',12321);

var TestOJO=require(‘react-native’).NativeModules.TestOJO;(将OC注册进来的模块取出)

TestOJO.j2oFun1(‘Himi’,12321);(调用模块中的对应函数,且将参数进行传入)

6. 我们来看一段复杂的数据通信

OC 代码段(导出函数):

#import"RCTConvert.h"
RCT_EXPORT_METHOD(j2oFun2:(NSDictionary*)details)
{
Nsstring*name=[RCTConvertNsstring:details[@"name"]];
NSNumber*age=[RCTConvertNSNumber:details[@"age"]];
NSArray*array=[RCTConvertNSArray:details[@"array"]];
NSLog(@"jscalliOSfunctionj2oFun2\nname:%@|age:%@",name,[agestringValue]);

for(inti=0;i<[arraycount];i++){
NSLog(@"array:第%d个元素:%@",i,array[i]);
}
}

需要注意的是,引入了”RCTConvert”类,作用:

RCTConvert提供了一系列辅助函数,用来接收一个JSON值并转换到原生Objective-C类型或类。

JS代码段:(调用OC函数

TestOJO.j2oFun2({
name:'Himi',age:12,array:[
'hi,Himi','i,m','aarray!'
]
});

7. 我们下面来利用回调参数来得到访问OC的函数得到其返回值

RCT_EXPORT_METHOD(j2oCallbackEvent:(Nsstring*)jsstringcallback:(RCTResponseSenderBlock)callback)
{
NSLog(@"jscalliOSfunction:j2oCallbackEvent\njsstring:%@",jsstring);
NSArray*events=[[NSArrayalloc]initWithObjects:@"Himi",@"12321",nil];
callback(@[[NSNullnull],events]);
}

RCTResponseSenderBlock 是种特殊的参数类型――回调函数,通过此参数可以实现当JS访问的OC函数后,并能将此OC函数的返回值传递给JS。

RCTResponseSenderBlock 只接受一个参数(传递给JavaScript回调函数的参数数组)

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

下面我们来看JS调用代码段:

TestOJO.j2oCallbackEvent('Himi',(error,callBackEvents)=>{
if(error){
console.error(error);
}else{
Alert.alert('J2O带返回值','数组的三个值:\n[0]:'+callBackEvents[0]+'\n[1]:'+callBackEvents[1]+'\n[2]:'+callBackEvents[2]);
}
});

二:iOS访问React Native

1. 我们如果想要OC访问JS,给JavaScript发送事件通知,我们需要使用RCTEventdispatcher的函数,与RCTBridge的实例

因此我们需要先做准备,TestOJO.h:

#import"RCTEventdispatcher.h"
@synthesizebridge=_bridge;

bridge: 是RCTBridge 的实例,且在我们使用的接口RCTBridgeModule中。

OC访问JS的代码段:

[self.bridge.eventdispatchersendAppEventWithName:@"eventName"body:@{@"name":@"Himi",@"age":@12}];

一个参数:事件名

第二个参数(body):传入的参数

其中@{}是定义不可变的字典的快捷实例方式,因此我们也可以改成如下形式:

NSDictionary*direct=@{@"name":@"Himi",@"age":@12};
[self.bridge.eventdispatchersendAppEventWithName:@"eventName"body:direct];

下面来看JS中定义OC调用函数

其实所谓OC能响应JS,是JS进行了对应函数的绑定监听。因此我们需要利用NativeAppEventEmitter 组件,利用其addListener进行注册监听!因此我们需要引入进来这个模块,

import{
...
NativeAppEventEmitter
...
}from'react-native';
varo2cFun=NativeAppEventEmitter.addListener(
'eventName',(para)=>Alert.alert('被OC触发','字典数据:\nname:'+para.name+'\nage:'+para.age)
);

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

addListener:

第二个参数:响应函数

注意:利用addListener进行监听,一定要对应有取消监听!要保持一一对应的好习惯。

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

componentwillUnmount(){
o2cFun.remove();
}

其中对于原理并没有详细的介绍,这里推荐两篇文章,童鞋们可以详细的阅读一下,这里不赘述:

http://www.jianshu.com/p/203b91a77174

http://reactnative.cn/docs/0.21/native-modules-ios.html#content

下面给出源码:

TestOJO.h:

//
//TestOJO.h
//MyProject
//
//CreatedbyHimion16/6/2.
//copyright2016年Facebook.Allrightsreserved.
//
#import<Foundation/Foundation.h>
#import"RCTBridgeModule.h"
@interfaceTestOJO:NSObject<RCTBridgeModule>
@end

TestOJO.m:

//
//TestOJO.m
//MyProject
//
//CreatedbyHimion16/6/2.
//copyright2016年Facebook.Allrightsreserved.
//
#import"TestOJO.h"
//RCTConvert类支持的的类型也都可以使用,RCTConvert还提供了一系列辅助函数,用来接收一个JSON值并转换到原生Objective-C类型或类。
#import"RCTConvert.h"
//本地模块也可以给JavaScript发送事件通知。最直接的方式是使用eventdispatcher
#import"RCTEventdispatcher.h"
@implementationTestOJO
//====================================[JS->OC]=======================================
RCT_EXPORT_MODULE();
//桥接到Javascript的方法返回值类型必须是void。ReactNative的桥接操作是异步的,所以要返回结果给Javascript,必须通过回调或者触发事件来进行
RCT_EXPORT_METHOD(j2oFun1:(Nsstring*)dataStringdateNumber:(int)dateNumber)
{
NSLog(@"jscalliOSfunctionj2oFun1\ndataString:%@|dateNumber:%d",dateNumber);
}
RCT_EXPORT_METHOD(j2oFun2:(NSDictionary*)details)
{
Nsstring*name=[RCTConvertNsstring:details[@"name"]];
NSNumber*age=[RCTConvertNSNumber:details[@"age"]];
NSArray*array=[RCTConvertNSArray:details[@"array"]];
NSLog(@"jscalliOSfunctionj2oFun2\nname:%@|age:%@",array[i]);
}

}
//带回调函数RCTResponseSenderBlock,提供将返回值传回给js
//RCTResponseSenderBlock只接受一个参数->传递给JavaScript回调函数的参数数组
RCT_EXPORT_METHOD(j2oCallbackEvent:(Nsstring*)jsstringcallback:(RCTResponseSenderBlock)callback)
{
NSLog(@"jscalliOSfunction:j2oCallbackEvent\njsstring:%@",events]);
}
//====================================[OC->JS]=======================================
@synthesizebridge=_bridge;
//此函数是为了测试OC->JS过程,触发事件的函数
RCT_EXPORT_METHOD(emitterO2J)
{
[selfocCallJsFun];
}
-(void)ocCallJsFun
{
NSDictionary*direct=@{@"name":@"Himi",@"age":@12};
[self.bridge.eventdispatchersendAppEventWithName:@"eventName"body:direct];

//[self.bridge.eventdispatchersendAppEventWithName:@"eventName"body:@{@"name":@"Himi",@"age":@12}];

}
@end


Main.js:

importReact,{Component}from'react';
import{
View,Text,StyleSheet,Image,Alert,NativeAppEventEmitter,//引用NativeAppEventEmitter组件进行监听Native端派发的事件
}from'react-native';
varTestOJO=require('react-native').NativeModules.TestOJO;
varo2cFun=NativeAppEventEmitter.addListener(
'eventName','字典数据:\nname:'+para.name+'\nage:'+para.age)
);
//千万不要忘记忘记取消订阅,通常在componentwillUnmount函数中实现。
//o2cFun.remove();
exportdefaultclassMainextendsComponent{
	constructor(props){
		super(props);
		this.state={
selectedTab:'home'
};
	}
componentwillUnmount(){
o2cFun.remove();
}
render(){
return(
	<Viewstyle={{flex:1,alignItems:'center'}}>
<Textstyle={styles.himiTextStyle}>HimiReactNative系列教程</Text>
<Text
onPress={()=>{
TestOJO.j2oFun1('Himi',12321);
TestOJO.j2oFun2({
name:'Himi','aarray!'
]
});
TestOJO.j2oCallbackEvent('Himi','数组的三个值:\n[0]:'+callBackEvents[0]+'\n[1]:'+callBackEvents[1]+'\n[2]:'+callBackEvents[2]);
}
});
}}
style={styles.himiTextStyle}>JS->OC
</Text>
<Text
onPress={()=>{
TestOJO.emitterO2J();
}}
style={styles.himiTextStyle}>JS->OC->JS
</Text>
</View>
);
}
};
varstyles=StyleSheet.create({
himiTextStyle:{
backgroundColor:'#eee',color:'#f00',fontSize:30,marginTop:70,},});

下面是运行效果:(点击看动态图,主要看演示过程与控制台输出哦!)

user918

注意:

1.点击JS->OC 后,会调用三个函数

2.点击JS->OC->JS,先是通过JS->OC的临时函数,触发OC->JS的过程!

相关文章

react 中的高阶组件主要是对于 hooks 之前的类组件来说的,如...
我们上一节了解了组件的更新机制,但是只是停留在表层上,例...
我们上一节了解了 react 的虚拟 dom 的格式,如何把虚拟 dom...
react 本身提供了克隆组件的方法,但是平时开发中可能很少使...
mobx 是一个简单可扩展的状态管理库,中文官网链接。小编在接...
我们在平常的开发中不可避免的会有很多列表渲染逻辑,在 pc ...