跳到主要内容

Headless JS

Headless JS 是一种在应用处于后台时运行 JavaScript 任务的方法。它可以用于例如同步最新数据、处理推送通知或播放音乐。

JS API

任务是一个异步函数,你需要在 AppRegistry 上注册它,类似于注册 React 应用:

tsx
import {AppRegistry} from 'react-native';
AppRegistry.registerHeadlessTask('SomeTaskName', () =>
require('SomeTaskName'),
);

然后,在 SomeTaskName.js 中:

tsx
module.exports = async taskData => {
// 执行任务
};

你可以在任务中执行任何操作,如网络请求、定时器等,只要不涉及 UI 操作。一旦任务完成(即 promise 被解决),React Native 将进入“暂停”模式(除非有其他任务在运行,或者有前台应用)。

平台 API

是的,这仍然需要一些原生代码,但非常简单。你需要继承 HeadlessJsTaskService 并重写 getTaskConfig,例如:

java
package com.your_application_name;

import android.content.Intent;
import android.os.Bundle;
import com.facebook.react.HeadlessJsTaskService;
import com.facebook.react.bridge.Arguments;
import com.facebook.react.jstasks.HeadlessJsTaskConfig;
import javax.annotation.Nullable;

public class MyTaskService extends HeadlessJsTaskService {

@Override
protected @Nullable HeadlessJsTaskConfig getTaskConfig(Intent intent) {
Bundle extras = intent.getExtras();
if (extras != null) {
return new HeadlessJsTaskConfig(
"SomeTaskName",
Arguments.fromBundle(extras),
5000, // 任务的超时时间,单位毫秒
false // 可选:定义任务是否允许在前台运行,默认是 false
);
}
return null;
}
}

然后,将服务添加到 AndroidManifest.xml 文件中的 application 标签内:

xml
<service android:name="com.example.MyTaskService" />

现在,每当你启动服务,例如作为定期任务或响应某些系统事件 / 广播时,JS 将启动,运行你的任务,然后关闭。

示例:

java
Intent service = new Intent(getApplicationContext(), MyTaskService.class);
Bundle bundle = new Bundle();

bundle.putString("foo", "bar");
service.putExtras(bundle);

getApplicationContext().startForegroundService(service);

重试

默认情况下,headless JS 任务不会执行任何重试。若要实现重试,你需要创建一个 HeadlessJsRetryPolicy 并抛出特定的 Error

LinearCountingRetryPolicyHeadlessJsRetryPolicy 的一个实现,允许你指定最大重试次数以及每次重试间的固定延迟。如果这不符合你的需求,则可以自定义实现 HeadlessJsRetryPolicy。这些策略可以作为额外参数传递给 HeadlessJsTaskConfig 构造函数,例如:

java
HeadlessJsRetryPolicy retryPolicy = new LinearCountingRetryPolicy(
3, // 最大重试次数
1000 // 每次重试间的延迟
);

return new HeadlessJsTaskConfig(
'SomeTaskName',
Arguments.fromBundle(extras),
5000,
false,
retryPolicy
);

只有当抛出特定的 Error 时才会尝试重试。在 headless JS 任务内部,可以导入该错误并在需要重试时抛出。

示例:

tsx
import {HeadlessJsTaskError} from 'HeadlessJsTask';

module.exports = async taskData => {
const condition = ...;
if (!condition) {
throw new HeadlessJsTaskError();
}
};

如果希望所有错误都触发重试,需要捕获错误并抛出上述错误。

注意事项

  • 默认情况下,如果你尝试在应用处于前台时运行任务,应用会崩溃。如此设计是为了防止开发者因在任务中执行大量工作而导致 UI 卡顿。你可以传入第四个布尔参数来控制此行为。
  • 如果你从 BroadcastReceiver 启动服务,确保在 onReceive() 返回之前调用 HeadlessJsTaskService.acquireWakeLockNow()

示例用法

服务可以通过 Java API 启动。首先需要决定何时启动服务,并据此实现方案。以下示例响应网络连接变化。

以下是 AndroidManifest 文件中注册广播接收器的部分内容:

xml
<receiver android:name=".NetworkChangeReceiver" >
<intent-filter>
<action android:name="android.net.conn.CONNECTIVITY_CHANGE" />
</intent-filter>
</receiver>

广播接收器处理广播的 intent,位于 onReceive 函数中。这是检查应用是否在前台的好地方。如果应用不在前台,可以准备要启动的 intent,附带无信息或使用 putExtra 打包的额外信息(注意 bundle 只能处理可序列化的值)。最后启动服务并获取 wakelock。

java
import android.app.ActivityManager;
import android.content.BroadcastReceiver;
import android.content.Context;
import android.content.Intent;
import android.net.ConnectivityManager;
import android.net.Network;
import android.net.NetworkCapabilities;
import android.net.NetworkInfo;
import android.os.Build;

import com.facebook.react.HeadlessJsTaskService;

public class NetworkChangeReceiver extends BroadcastReceiver {

@Override
public void onReceive(final Context context, final Intent intent) {
/**
每当网络连接发生变化时调用
例如:已连接 -> 未连接
**/
if (!isAppOnForeground((context))) {
/**
我们将启动服务并发送网络连接的额外信息
**/
boolean hasInternet = isNetworkAvailable(context);
Intent serviceIntent = new Intent(context, MyTaskService.class);
serviceIntent.putExtra("hasInternet", hasInternet);
context.startForegroundService(serviceIntent);
HeadlessJsTaskService.acquireWakeLockNow(context);
}
}

private boolean isAppOnForeground(Context context) {
/**
需要检查应用是否在前台,否则应用会崩溃。
https://stackoverflow.com/questions/8489993/check-android-application-is-in-foreground-or-not
**/
ActivityManager activityManager = (ActivityManager) context.getSystemService(Context.ACTIVITY_SERVICE);
List<ActivityManager.RunningAppProcessInfo> appProcesses =
activityManager.getRunningAppProcesses();
if (appProcesses == null) {
return false;
}
final String packageName = context.getPackageName();
for (ActivityManager.RunningAppProcessInfo appProcess : appProcesses) {
if (appProcess.importance ==
ActivityManager.RunningAppProcessInfo.IMPORTANCE_FOREGROUND &&
appProcess.processName.equals(packageName)) {
return true;
}
}
return false;
}

public static boolean isNetworkAvailable(Context context) {
ConnectivityManager cm = (ConnectivityManager)
context.getSystemService(Context.CONNECTIVITY_SERVICE);

if (Build.VERSION.SDK_INT >= Build.VERSION_CODES.M) {
Network networkCapabilities = cm.getActiveNetwork();

if(networkCapabilities == null) {
return false;
}

NetworkCapabilities actNw = cm.getNetworkCapabilities(networkCapabilities);

if(actNw == null) {
return false;
}

if(actNw.hasTransport(NetworkCapabilities.TRANSPORT_WIFI) || actNw.hasTransport(NetworkCapabilities.TRANSPORT_CELLULAR) || actNw.hasTransport(NetworkCapabilities.TRANSPORT_ETHERNET)) {
return true;
}

return false;
}

// API 29 中已废弃
NetworkInfo netInfo = cm.getActiveNetworkInfo();
return (netInfo != null && netInfo.isConnected());
}
}