Android模块化及RN调用原生
先老实交代下背景,这篇讲的是两个问题。这两个问题是自己被问到的,但是讲的很含糊,想必问我的人也不满意吧。究其原因:语言组织、表达、紧张。都是泪,下面说说这两个问题吧。
Q1:android模块化遇到的问题?
按照原始MVP的架构,随着业务越来越复杂、基础组件越来越多。业务与业务之间还有很强的耦合。就变成这样
复杂业务使用模块化之后,变成这样
模块化很明显清晰了不少。那么模块化又存在哪些问题呢?
- 最常见的R文件问题。
我当时说的时候:整体与个体之间冲突,findbyid常用的ButterKnife不能用,因为会有冲突等。module的application和主app的application冲突。
总之没有很系统的讲出来,估计对方也不耐烦了。尴了个噶。接下来正规系统的讲下:
Android开发中,各个资源文件都是放在res目录中,在编译过程中,会生成R.java文件。R文件中包含有各个资源文件对应的id,这个id是静态常量,但是在Library Module中,这个id不是静态常量,那么就会出现找不到id或者id冲突这样的问题。因此常用的ButterKnite也不能使用
解决方案:
- 重新一个Gradle插件,生成一个R2.java文件,这个文件中各个id都是静态常量,这样ButterKnite就可以正常使用了,通过R2获取到id。
- 使用Android系统提供的最原始的方式,直接用findViewById以及setOnClickListener方式。
- 设置项目支持Databinding,然后使用Binding中的对象,但是会增加不少方法数,同时Databinding也会有编译问题和学习成本。目前项目中没有使用Databinging,这里也没有太多的发言权
- xml冲突,资源冲突问题
解决办法:
- 对各个模块的资源名字添加前缀,比如user模块中的登录界面布局为activity_login.xml,那么可以写成这样us_activity_login.xml。
- application的xml只能查查哪些属性冲突,借助网上资料了。
总结
模块化架构主要思路就是分而治之,把依赖整理清楚,减少代码冗余和耦合,在把代码抽取到各自的模块后,了解各个模块的通信方式,以及可能发生的问题,规避问题或者解决问题。最后为了开发和调试方便,开发一些周边工具,帮助开发更好的完成任务。
Q2 ReactNative如何调用Android原生模块
先来说说我的回答:android原生模块实现ReactContext,通过它的回调使用原生模块的方法。现在回想太通俗了。下面正规说下:
- 创建一个原生模块
首先我们需要创建一个原生模块,这个原生模块是一个继承ReactContextBaseJavaModule的Java类,它可以实现一些JavaScript所调用的原生功能.
public class RnTest extends ReactContextBaseJavaModule {
public RnTest(ReactApplicationContext reactContext) {
super(reactContext);
}
// ReactContextBaseJavaModule要求派生类实现getName方法。这个函数用于返回一个字符串
// 这个字符串用于在JavaScript端标记这个原生模块
@Override
public String getName() {
return "ToastByAndroid";
}
// 获取应用包名
// 要导出一个方法给JavaScript使用,Java方法需要使用注解@ReactMethod
@ReactMethod
public void getPackageName() {
String name = getReactApplicationContext().getPackageName();
Toast.makeText(getReactApplicationContext(),name,Toast.LENGTH_LONG).show();
}
}
- 注册模块
要使JavaScript端调用到原生模块还需注册这个原生模块。需实现一个类实现ReactPackage接口
public class ExampleReactPackage implements ReactPackage {
@Override
public List<NativeModule> createNativeModules(ReactApplicationContext reactContext) {
List<NativeModule> modules = new ArrayList<>();
modules.add(new RnTest(reactContext));
return modules;
}
@Override
public List<Class<? extends JavaScriptModule>> createJSModules() {
return Collections.emptyList();
}
@Override
public List<ViewManager> createViewManagers(ReactApplicationContext reactContext) {
return Collections.emptyList();
}
}
还需在MainApplication.java文件中的getPackages方法中,实例化上面的注册类
@Override
protected List<ReactPackage> getPackages() {
return Arrays.<ReactPackage>asList(
new MainReactPackage(),
// 实例化注册类
new ExampleReactPackage());
}
};
-
js调用android原生
- 引入模块
import { NativeModules } from 'react-native';
- 使用方法
- 引入模块
// 这里的ToastByAndroid即为1.创建一个原生模块中getName()方法返回的字符串
var rnToastAndroid = NativeModules.ToastByAndroid;
rnToastAndroid.getPackageName();
3. 回调函数
提供给js调用的原生android方法的返回类型必须是void,React Native的跨语言访问是异步进行的,所以想要给JavaScript返回一个值的唯一办法是使用回调函数或者发送事件
android主动向rn发送消息
既然都说道js向原生调用了,那么也说说原生向js调用。
根据上面,可以想到的思路就是:通过reactContext获取到jsModule,然后像eventbus一样发生事件,js方获取事件。
- 创建一个原生模块,发送消息方法
public static void sendEvent(ReactContext reactContext, String eventName, int status)
{
System.out.println("reactContext="+reactContext);
reactContext
.getJSModule(DeviceEventManagerModule.RCTDeviceEventEmitter.class)
.emit(eventName,status);
}
```
2. RN端代码
DeviceEventEmitter.addListener(eventName, (reminder) => {
console.log(reminder):
});
```
- rn调用android模版
const RNBridgeModule = NativeModules.RNBridgeModule;
nativeLanuchApp(message) {
RNBridgeModule.nativePlayVideo(message);
}
<TouchableOpacity onPress={() => {
this.nativeLanuchApp("111");
}} >
<Text >
try
</Text>
</TouchableOpacity>
```
网友评论