跳到主要内容
版本:0.79

Android 原生模块

信息

原生模块和原生组件是我们旧架构中使用的稳定技术。 当新架构稳定后,它们将在未来被弃用。新架构使用 Turbo Native ModuleFabric Native Components 来实现类似的结果。

欢迎使用 Android 原生模块。请先阅读 原生模块简介 以了解什么是原生模块。

创建日历原生模块

在以下指南中,您将创建一个原生模块,CalendarModule,它将允许您从 JavaScript 访问 Android 的日历 API。最后,您将能够从 JavaScript 调用 CalendarModule.createCalendarEvent('Dinner Party', 'My House');,调用一个创建日历事件的 Java/Kotlin 方法。

设置

首先,在 Android Studio 中打开 React Native 应用内的 Android 项目。您可以在 React Native 应用中的此处找到您的 Android 项目:

在 Android Studio 内的 React Native 应用中打开 Android 项目的图像。
您可以在哪里找到 Android 项目的图像

我们建议使用 Android Studio 编写原生代码。Android Studio 是为 Android 开发构建的 IDE,使用它将帮助您快速解决代码语法错误等小问题。

我们还建议启用 Gradle Daemon 以在您迭代 Java/Kotlin 代码时加快构建速度。

创建自定义原生模块文件

第一步是在 android/app/src/main/java/com/your-app-name/ 文件夹内创建 (CalendarModule.javaCalendarModule.kt) Java/Kotlin 文件(Kotlin 和 Java 的文件夹相同)。此 Java/Kotlin 文件将包含您的原生模块 Java/Kotlin 类。

在 Android Studio 内添加名为 CalendarModule.java 的类的图像。
如何添加 CalendarModuleClass 的图像

然后添加以下内容:

java
package com.your-apps-package-name; // 将 your-apps-package-name 替换为您应用的包名
import com.facebook.react.bridge.NativeModule;
import com.facebook.react.bridge.ReactApplicationContext;
import com.facebook.react.bridge.ReactContext;
import com.facebook.react.bridge.ReactContextBaseJavaModule;
import com.facebook.react.bridge.ReactMethod;
import java.util.Map;
import java.util.HashMap;

public class CalendarModule extends ReactContextBaseJavaModule {
CalendarModule(ReactApplicationContext context) {
super(context);
}
}

如您所见,您的 CalendarModule 类扩展了 ReactContextBaseJavaModule 类。对于 Android,Java/Kotlin 原生模块编写为扩展 ReactContextBaseJavaModule 并实现 JavaScript 所需功能的类。

值得注意的是,技术上 Java/Kotlin 类只需要扩展 BaseJavaModule 类或实现 NativeModule 接口即可被 React Native 视为原生模块。

但是,我们建议您使用 ReactContextBaseJavaModule,如上所示。ReactContextBaseJavaModule 提供对 ReactApplicationContext (RAC) 的访问,这对于需要挂钩活动生命周期方法的原生模块很有用。使用 ReactContextBaseJavaModule 还将使将来更容易使您的原生模块类型安全。对于原生模块类型安全(将在未来版本中推出),React Native 会查看每个原生模块的 JavaScript 规范,并生成一个扩展 ReactContextBaseJavaModule 的抽象基类。

模块名称

Android 中的所有 Java/Kotlin 原生模块都需要实现 getName() 方法。此方法返回一个字符串,代表原生模块的名称。然后可以在 JavaScript 中使用其名称访问原生模块。例如,在下面的代码片段中,getName() 返回 "CalendarModule"

java
// 添加到 CalendarModule.java
@Override
public String getName() {
return "CalendarModule";
}

然后可以在 JS 中这样访问原生模块:

tsx
const {CalendarModule} = ReactNative.NativeModules;

向 JavaScript 导出原生方法

接下来,您需要向原生模块添加一个方法,该方法将创建日历事件并且可以在 JavaScript 中调用。所有旨在从 JavaScript 调用的原生模块方法必须用 @ReactMethod 注解。

CalendarModule 设置一个 createCalendarEvent() 方法,该方法可以通过 CalendarModule.createCalendarEvent() 在 JS 中调用。目前,该方法将接受名称和位置作为字符串。参数类型选项将在稍后介绍。

java
@ReactMethod
public void createCalendarEvent(String name, String location) {
}

在方法中添加调试日志以确认在从应用程序调用时已调用它。下面是如何从 Android util 包导入和使用 Log 类的示例:

java
import android.util.Log;

@ReactMethod
public void createCalendarEvent(String name, String location) {
Log.d("CalendarModule", "Create event called with name: " + name
+ " and location: " + location);
}

完成原生模块的实现并在 JavaScript 中连接后,您可以遵循 这些步骤 查看应用中的日志。

同步方法

您可以将 isBlockingSynchronousMethod = true 传递给原生方法以将其标记为同步方法。

java
@ReactMethod(isBlockingSynchronousMethod = true)

目前,我们不推荐这样做,因为同步调用方法可能会产生严重的性能惩罚,并向您的原生模块引入与线程相关的错误。此外,请注意,如果您选择启用 isBlockingSynchronousMethod,您的应用将无法再使用 Google Chrome 调试器。这是因为同步方法需要 JS VM 与应用共享内存。对于 Google Chrome 调试器,React Native 在 Google Chrome 的 JS VM 内运行,并通过 WebSockets 与移动设备异步通信。

注册模块(Android 特定)

编写完原生模块后,需要将其注册到 React Native。为此,您需要将原生模块添加到 ReactPackage 并将 ReactPackage 注册到 React Native。在初始化期间,React Native 将遍历所有包,并为每个 ReactPackage 注册其中的每个原生模块。

React Native 在 ReactPackage 上调用 createNativeModules() 方法以获取要注册的原生模块列表。对于 Android,如果模块未在 createNativeModules 中实例化并返回,则无法从 JavaScript 访问。

要将您的原生模块添加到 ReactPackage,首先在 android/app/src/main/java/com/your-app-name/ 文件夹内创建一个实现 ReactPackage 的新 Java/Kotlin 类,名为 (MyAppPackage.javaMyAppPackage.kt):

然后添加以下内容:

java
package com.your-app-name; // 将 your-app-name 替换为您应用的名称
import com.facebook.react.ReactPackage;
import com.facebook.react.bridge.NativeModule;
import com.facebook.react.bridge.ReactApplicationContext;
import com.facebook.react.uimanager.ViewManager;

import java.util.ArrayList;
import java.util.Collections;
import java.util.List;

public class MyAppPackage implements ReactPackage {

@Override
public List<ViewManager> createViewManagers(ReactApplicationContext reactContext) {
return Collections.emptyList();
}

@Override
public List<NativeModule> createNativeModules(
ReactApplicationContext reactContext) {
List<NativeModule> modules = new ArrayList<>();

modules.add(new CalendarModule(reactContext));

return modules;
}

}

此文件导入您创建的原生模块,CalendarModule。然后它在 createNativeModules() 函数内实例化 CalendarModule 并将其作为 NativeModules 列表返回以进行注册。如果您以后添加更多原生模块,也可以实例化它们并将它们添加到此处返回的列表中。

值得注意的是,这种注册原生模块的方式会在应用启动时急切地初始化所有原生模块,这会增加应用的启动时间。您可以使用 TurboReactPackage 作为替代方案。TurboReactPackage 不返回实例化的原生模块对象列表的 createNativeModules,而是实现一个 getModule(String name, ReactApplicationContext rac) 方法,该方法在需要时创建原生模块对象。目前实现 TurboReactPackage 有点复杂。除了实现 getModule() 方法外,您还必须实现 getReactModuleInfoProvider() 方法,该方法返回包可以实例化的所有原生模块列表以及实例化它们的函数,示例 此处。同样,使用 TurboReactPackage 将使您的应用具有更快的启动时间,但目前编写起来有点麻烦。因此,如果您选择使用 TurboReactPackages,请谨慎行事。

要注册 CalendarModule 包,您必须将 MyAppPackage 添加到 ReactNativeHost 的 getPackages() 方法返回的包列表中。打开 MainApplication.javaMainApplication.kt 文件,可以在以下路径找到:android/app/src/main/java/com/your-app-name/

找到 ReactNativeHost 的 getPackages() 方法,并将您的包添加到 getPackages() 返回的包列表中:

java
@Override
protected List<ReactPackage> getPackages() {
List<ReactPackage> packages = new PackageList(this).getPackages();
// 尚未自动链接的包可以手动添加到这里,例如:
// packages.add(new MyReactNativePackage());
packages.add(new MyAppPackage());
return packages;
}

您现在已成功为 Android 注册了原生模块!

测试您的构建成果

此时,您已经在 Android 中为原生模块设置了基本框架。通过在 JavaScript 中访问原生模块并调用其导出方法来测试一下。

在应用程序中找到想要添加对原生模块 createCalendarEvent() 方法调用的位置。下面是您可以在应用中添加的组件,NewModuleButton 的示例。您可以在 NewModuleButtononPress() 函数内调用原生模块。

tsx
import React from 'react';
import {NativeModules, Button} from 'react-native';

const NewModuleButton = () => {
const onPress = () => {
console.log('We will invoke the native module here!');
};

return (
<Button
title="Click to invoke your native module!"
color="#841584"
onPress={onPress}
/>
);
};

export default NewModuleButton;

为了从 JavaScript 访问您的原生模块,您需要首先从 React Native 导入 NativeModules

tsx
import {NativeModules} from 'react-native';

然后您可以从 NativeModules 访问 CalendarModule 原生模块。

tsx
const {CalendarModule} = NativeModules;

现在您有了 CalendarModule 原生模块,您可以调用您的原生方法 createCalendarEvent()。下面将其添加到 NewModuleButton 中的 onPress() 方法:

tsx
const onPress = () => {
CalendarModule.createCalendarEvent('testName', 'testLocation');
};

最后一步是重新构建 React Native 应用,以便您可以使用最新的原生代码(包含您的新原生模块!)。在命令行中,在 react native 应用所在的位置,运行以下命令:

npm run android

迭代构建

当您通过这些指南并迭代原生模块时,您需要对应用程序进行原生重建,以便从 JavaScript 访问您的最新更改。这是因为您编写的代码位于应用程序的原生部分内。虽然 React Native 的 metro bundler 可以监视 JavaScript 中的更改并为您即时重建,但它不会为原生代码这样做。因此,如果您想测试最新的原生更改,需要使用上述命令重新构建。

回顾✨

您现在应该能够在应用中调用原生模块上的 createCalendarEvent() 方法。在我们的示例中,这是通过按下 NewModuleButton 发生的。您可以通过查看在 createCalendarEvent() 方法中设置的日志来确认这一点。您可以遵循 这些步骤 查看应用中的 ADB 日志。然后您应该能够搜索您的 Log.d 消息(在我们的示例中为"Create event called with name: testName and location: testLocation"),并看到每次调用原生模块方法时记录的消息。

日志图像。
Android Studio 中 ADB 日志的图像

此时,您已创建一个 Android 原生模块,并在 React Native 应用中从 JavaScript 调用了其原生方法。您可以继续阅读以了解更多内容,例如原生模块方法可用的参数类型以及如何设置回调和 Promise。

超越 Calendar 原生模块

更好的原生模块导出

像上面那样通过从 NativeModules 中提取来导入你的原生模块有点笨拙。

为了节省你的原生模块使用者每次访问原生模块时都需要这样做的麻烦,你可以为该模块创建一个 JavaScript 包装器。创建一个名为 CalendarModule.js 的新 JavaScript 文件,内容如下:

tsx
/**
* 这将原生 CalendarModule 模块暴露为 JS 模块。它有一个
* 函数 'createCalendarEvent',接受以下参数:

* 1. String name: 代表事件名称的字符串
* 2. String location: 代表事件地点的字符串
*/
import {NativeModules} from 'react-native';
const {CalendarModule} = NativeModules;
export default CalendarModule;

这个 JavaScript 文件也成为你添加任何 JavaScript 端功能的好地方。例如,如果你使用像 TypeScript 这样的类型系统,你可以在这里为你的原生模块添加类型注解。虽然 React Native 尚未支持原生到 JS 的类型安全,但你所有的 JS 代码将是类型安全的。这样做也将使你更容易在未来切换到类型安全的原生模块。下面是为 CalendarModule 添加类型安全的示例:

tsx
/**
* 这将原生 CalendarModule 模块暴露为 JS 模块。它有一个
* 函数 'createCalendarEvent',接受以下参数:
*
* 1. String name: 代表事件名称的字符串
* 2. String location: 代表事件地点的字符串
*/
import {NativeModules} from 'react-native';
const {CalendarModule} = NativeModules;
interface CalendarInterface {
createCalendarEvent(name: string, location: string): void;
}
export default CalendarModule as CalendarInterface;

在你的其他 JavaScript 文件中,你可以访问原生模块并像这样调用其方法:

tsx
import CalendarModule from './CalendarModule';
CalendarModule.createCalendarEvent('foo', 'bar');

这假设你导入 CalendarModule 的位置与 CalendarModule.js 处于相同的层级结构中。请根据需要更新相对导入。

参数类型

当在 JavaScript 中调用原生模块方法时,React Native 会将参数从 JS 对象转换为其 Java/Kotlin 对象 аналог。例如,如果你的 Java 原生模块方法接受一个 double,在 JS 中你需要用数字调用该方法。React Native 会为你处理转换。下面是原生模块方法支持的参数类型列表及其映射的 JavaScript 等效类型。

JavaKotlinJavaScript
BooleanBoolean?boolean
booleanboolean
DoubleDouble?number
doublenumber
StringStringstring
CallbackCallbackFunction
PromisePromisePromise
ReadableMapReadableMapObject
ReadableArrayReadableArrayArray

以下类型目前受支持,但在 TurboModules 中将不再受支持。请避免使用它们:

  • Integer Java/Kotlin -> ?number
  • Float Java/Kotlin -> ?number
  • int Java -> number
  • float Java -> number

对于上面未列出的参数类型,你需要自己处理转换。例如,在 Android 中,Date 转换不支持开箱即用。你可以像这样在原生方法内处理到 Date 类型的转换:

java
    String dateFormat = "yyyy-MM-dd";
SimpleDateFormat sdf = new SimpleDateFormat(dateFormat);
Calendar eStartDate = Calendar.getInstance();
try {
eStartDate.setTime(sdf.parse(startDate));
}

导出常量

原生模块可以通过实现原生方法 getConstants() 来导出常量,该方法在 JS 中可用。下面你将实现 getConstants() 并返回一个包含 DEFAULT_EVENT_NAME 常量的 Map,你可以在 JavaScript 中访问它:

java
@Override
public Map<String, Object> getConstants() {
final Map<String, Object> constants = new HashMap<>();
constants.put("DEFAULT_EVENT_NAME", "New Event");
return constants;
}

然后可以通过在 JS 中的原生模块上调用 getConstants 来访问常量:

tsx
const {DEFAULT_EVENT_NAME} = CalendarModule.getConstants();
console.log(DEFAULT_EVENT_NAME);

从技术上讲,可以直接从原生模块对象访问 getConstants() 中导出的常量。这在 TurboModules 中将不再受支持,所以我们鼓励社区切换到上述方法,以避免将来必要的迁移。

目前常量仅在初始化时导出,因此如果你在运行时更改 getConstants 值,它不会影响 JavaScript 环境。这将随着 Turbomodules 而改变。使用 Turbomodules 时,getConstants() 将成为一个常规的原生模块方法,每次调用都会触及原生端。

回调

原生模块还支持一种独特的参数类型:回调。回调用于将数据从 Java/Kotlin 传递给 JavaScript 以用于异步方法。它们也可用于从原生端异步执行 JavaScript。

为了创建带有回调的原生模块方法,首先导入 Callback 接口,然后在你的原生模块方法中添加一个类型为 Callback 的新参数。回调参数有一些细微差别,这些很快将随着 TurboModules 而消除。首先,你的函数参数中只能有两个回调 - 一个 successCallback 和一个 failureCallback。此外,原生模块方法调用的最后一个参数,如果是函数,则被视为 successCallback,原生模块方法调用的倒数第二个参数,如果是函数,则被视为 failure callback。

java
import com.facebook.react.bridge.Callback;

@ReactMethod
public void createCalendarEvent(String name, String location, Callback callBack) {
}

你可以在你的 Java/Kotlin 方法中调用回调,提供你想传递给 JavaScript 的任何数据。请注意,你只能将可序列化的数据从原生代码传递给 JavaScript。如果你需要返回一个原生对象,你可以使用 WriteableMaps,如果你需要使用集合,使用 WritableArrays。同样重要的是要强调,回调不会在原生函数完成后立即调用。下面将在早期调用中创建的事件的 ID 传递给回调。

java
  @ReactMethod
public void createCalendarEvent(String name, String location, Callback callBack) {
Integer eventId = ...
callBack.invoke(eventId);
}

然后可以在 JavaScript 中使用以下方法访问此方法:

tsx
const onPress = () => {
CalendarModule.createCalendarEvent(
'Party',
'My House',
eventId => {
console.log(`Created a new event with id ${eventId}`);
},
);
};

另一个需要注意的重要细节是,原生模块方法只能调用一个回调,一次。这意味着你可以调用成功回调或失败回调,但不能同时调用,并且每个回调最多只能调用一次。然而,原生模块可以存储回调并在以后调用它。

有两种使用回调进行错误处理的方法。第一种是遵循 Node 的约定,将传递给回调的第一个参数视为错误对象。

java
  @ReactMethod
public void createCalendarEvent(String name, String location, Callback callBack) {
Integer eventId = ...
callBack.invoke(null, eventId);
}

在 JavaScript 中,你可以检查第一个参数以查看是否传递了错误:

tsx
const onPress = () => {
CalendarModule.createCalendarEvent(
'testName',
'testLocation',
(error, eventId) => {
if (error) {
console.error(`Error found! ${error}`);
}
console.log(`event id ${eventId} returned`);
},
);
};

另一个选项是使用 onSuccess 和 onFailure 回调:

java
@ReactMethod
public void createCalendarEvent(String name, String location, Callback myFailureCallback, Callback mySuccessCallback) {
}

然后在 JavaScript 中你可以为错误和成功响应添加单独的回调:

tsx
const onPress = () => {
CalendarModule.createCalendarEvent(
'testName',
'testLocation',
error => {
console.error(`Error found! ${error}`);
},
eventId => {
console.log(`event id ${eventId} returned`);
},
);
};

Promises

原生模块也可以履行 Promise,这可以简化你的 JavaScript,特别是当使用 ES2016 的 async/await 语法时。当原生模块 Java/Kotlin 方法的最后一个参数是 Promise 时,其对应的 JS 方法将返回一个 JS Promise 对象。

将上述代码重构为使用 Promise 而不是回调如下所示:

java
import com.facebook.react.bridge.Promise;

@ReactMethod
public void createCalendarEvent(String name, String location, Promise promise) {
try {
Integer eventId = ...
promise.resolve(eventId);
} catch(Exception e) {
promise.reject("Create Event Error", e);
}
}

与回调类似,原生模块方法可以拒绝或解决一个 promise(但不能同时),并且最多只能这样做一次。这意味着你可以调用成功回调或失败回调,但不能同时调用,并且每个回调最多只能调用一次。然而,原生模块可以存储回调并在以后调用它。

此方法的 JavaScript 对应部分返回一个 Promise。这意味着你可以在异步函数中使用 await 关键字来调用它并等待其结果:

tsx
const onSubmit = async () => {
try {
const eventId = await CalendarModule.createCalendarEvent(
'Party',
'My House',
);
console.log(`Created a new event with id ${eventId}`);
} catch (e) {
console.error(e);
}
};

reject 方法接受以下参数的不同组合:

java
String code, String message, WritableMap userInfo, Throwable throwable

有关更多详细信息,你可以在这里找到 Promise.java 接口。如果未提供 userInfo,ReactNative 将其设置为 null。对于其余参数,React Native 将使用默认值。message 参数提供错误调用堆栈顶部显示的错误 message。下面是 Java/Kotlin 中以下 reject 调用在 JavaScript 中显示的错误消息示例。

Java/Kotlin reject 调用:

java
promise.reject("Create Event error", "Error parsing date", e);

Promise 被拒绝时 React Native App 中的错误消息:

React Native 应用中错误消息的图片
错误消息图片

向 JavaScript 发送事件

原生模块可以在不被直接调用的情况下向 JavaScript 发送事件信号。例如,你可能想要向 JavaScript 发送信号,提醒原生 Android 日历应用中的日历事件即将发生。最简单的方法是使用 RCTDeviceEventEmitter,可以从 ReactContext 获取,如下面的代码片段所示。

java
...
import com.facebook.react.modules.core.DeviceEventManagerModule;
import com.facebook.react.bridge.WritableMap;
import com.facebook.react.bridge.Arguments;
...
private void sendEvent(ReactContext reactContext,
String eventName,
@Nullable WritableMap params) {
reactContext
.getJSModule(DeviceEventManagerModule.RCTDeviceEventEmitter.class)
.emit(eventName, params);
}

private int listenerCount = 0;

@ReactMethod
public void addListener(String eventName) {
if (listenerCount == 0) {
// 设置任何上游监听器或后台任务(如有必要)
}

listenerCount += 1;
}

@ReactMethod
public void removeListeners(Integer count) {
listenerCount -= count;
if (listenerCount == 0) {
// 移除上游监听器,停止不必要的后台任务
}
}
...
WritableMap params = Arguments.createMap();
params.putString("eventProperty", "someValue");
...
sendEvent(reactContext, "EventReminder", params);

然后 JavaScript 模块可以通过 NativeEventEmitter 类上的 addListener 注册接收事件。

tsx
import {NativeEventEmitter, NativeModules} from 'react-native';
...
useEffect(() => {
const eventEmitter = new NativeEventEmitter(NativeModules.ToastExample);
let eventListener = eventEmitter.addListener('EventReminder', event => {
console.log(event.eventProperty) // "someValue"
});

// 卸载后移除监听器
return () => {
eventListener.remove();
};
}, []);

从 startActivityForResult 获取 Activity 结果

如果你想从使用 startActivityForResult 启动的 activity 获取结果,你需要监听 onActivityResult。为此,你必须扩展 BaseActivityEventListener 或实现 ActivityEventListener。前者是首选,因为它对 API 更改更具弹性。然后,你需要像这样在模块的构造函数中注册监听器:

java
reactContext.addActivityEventListener(mActivityResultListener);

现在你可以通过实现以下方法来监听 onActivityResult

java
@Override
public void onActivityResult(
final Activity activity,
final int requestCode,
final int resultCode,
final Intent intent) {
// 你的逻辑在这里
}

让我们实现一个基本的图像选择器来演示这一点。图像选择器将向 JavaScript 暴露 pickImage 方法,调用时将返回图像路径。

kotlin
public class ImagePickerModule extends ReactContextBaseJavaModule {

private static final int IMAGE_PICKER_REQUEST = 1;
private static final String E_ACTIVITY_DOES_NOT_EXIST = "E_ACTIVITY_DOES_NOT_EXIST";
private static final String E_PICKER_CANCELLED = "E_PICKER_CANCELLED";
private static final String E_FAILED_TO_SHOW_PICKER = "E_FAILED_TO_SHOW_PICKER";
private static final String E_NO_IMAGE_DATA_FOUND = "E_NO_IMAGE_DATA_FOUND";

private Promise mPickerPromise;

private final ActivityEventListener mActivityEventListener = new BaseActivityEventListener() {

@Override
public void onActivityResult(Activity activity, int requestCode, int resultCode, Intent intent) {
if (requestCode == IMAGE_PICKER_REQUEST) {
if (mPickerPromise != null) {
if (resultCode == Activity.RESULT_CANCELED) {
mPickerPromise.reject(E_PICKER_CANCELLED, "Image picker was cancelled");
} else if (resultCode == Activity.RESULT_OK) {
Uri uri = intent.getData();

if (uri == null) {
mPickerPromise.reject(E_NO_IMAGE_DATA_FOUND, "No image data found");
} else {
mPickerPromise.resolve(uri.toString());
}
}

mPickerPromise = null;
}
}
}
};

ImagePickerModule(ReactApplicationContext reactContext) {
super(reactContext);

// 添加 `onActivityResult` 的监听器
reactContext.addActivityEventListener(mActivityEventListener);
}

@Override
public String getName() {
return "ImagePickerModule";
}

@ReactMethod
public void pickImage(final Promise promise) {
Activity currentActivity = getCurrentActivity();

if (currentActivity == null) {
promise.reject(E_ACTIVITY_DOES_NOT_EXIST, "Activity doesn't exist");
return;
}

// 存储 promise 以便在选择器返回数据时解决/拒绝
mPickerPromise = promise;

try {
final Intent galleryIntent = new Intent(Intent.ACTION_PICK);

galleryIntent.setType("image/*");

final Intent chooserIntent = Intent.createChooser(galleryIntent, "Pick an image");

currentActivity.startActivityForResult(chooserIntent, IMAGE_PICKER_REQUEST);
} catch (Exception e) {
mPickerPromise.reject(E_FAILED_TO_SHOW_PICKER, e);
mPickerPromise = null;
}
}
}

监听生命周期事件

监听 activity 的生命周期事件(如 onResumeonPause 等)与实现 ActivityEventListener 非常相似。模块必须实现 LifecycleEventListener。然后,你需要像这样在模块的构造函数中注册监听器:

java
reactContext.addLifecycleEventListener(this);

现在你可以通过实现以下方法来监听 activity 的生命周期事件:

java
@Override
public void onHostResume() {
// Activity `onResume`
}
@Override
public void onHostPause() {
// Activity `onPause`
}
@Override
public void onHostDestroy() {
// Activity `onDestroy`
}

线程

迄今为止,在 Android 上,所有原生模块异步方法都在一个线程上执行。原生模块不应对其被调用的线程有任何假设,因为当前的分配在未来可能会发生变化。如果需要阻塞调用,应将繁重的工作分派给内部管理的工作线程,并从那里分发任何回调。