1. 写在前面 有J友在掘金私信我,react native android中,app在后台如何持续获取位置信息,还有headless js中setTimeout没有按预期执行两个问题。问我有什么解决方法,当时我就懵逼了,这不是触及到我装X盲区了吗,况且我只是js菜鸡,不会android,难受!
2. 本文主要 package version
package
version
react
18.2.0
react-native
0.71.2
@react-native-community/geolocation
^3.0.5
3. 前置基础
React 基础
React Native Android 原生模块 ,已经跟着文档,在js中调用android暴露的方法
4. 初步了解Headless JS
Headless JS文档
Headless JS 是一种使用 js 在后台执行任务的方法。它可以用来在后台同步数据、处理推送通知或是播放音乐等等。
可以在任务中处理任何事情(网络请求、定时器等),但**不要涉及UI界面
**
The function passed to setTimeout does not always behave as expected. Instead the function is called only when the application is launched again. If you just need to wait, use the retry functionality
,文档这里已经说明,headless js中setTimeout
不会按预期执行,而是会在app再次启动的时候才执行(就是app切到后台时,不会执行,切回前台的时候才执行),那用什么代替setTimout呢?下面会讲到。
Headless JS中发起网络请求,经过实际测试,完全没问题的
还有,app进程被杀掉(人为主动杀掉和系统资源优化掉),Headless JS后台任务也会停止 ,这里不讨论进程被杀掉还能继续执行后台任务
5. 使用Headless JS的姿势
在React Native 练习时长 2 月半,踩坑总结 文章中有涉及到使用Headless JS后台播放raw本地音频文件,那里是使用AppRegistry.startHeadlessTask(taskId, taskKey, data)
api开始后台任务的,在官方文档中有提到在service
中启动,但是步骤都不是非常详细
5.1 使用AppRegistry.startHeadlessTask
api启动Headless js后台任务 具体步骤,详见这篇文章-7.4章节app后台播放音频示例步骤 ,每一步都很详细,对着步骤来。
5.2 通过android WorkManager
中调用services
,启动Headless js后台任务
怎么突然又冒出来WorkManager
了?没办法啊,按着文档那种方式来,Headless JS中代码不执行,下面步骤1代码中会提到
WokerManager
是什么?
WorkManager is the recommended way to perform background tasks in Android. WorkManager can schedule one-time or periodic tasks in a simple, reliable way.
意思就是说,WorkManager是android中推荐执行后台任务的方式,可以执行一次性任务和定时任务。
android/app/src/main/java/com/your-app-name/BackgroundPosition.java
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 32 33 34 35 36 37 38 39 40 41 42 43 44 45 46 47 48 49 50 51 52 53 54 55 56 57 58 59 60 61 62 63 64 65 66 67 68 69 70 71 72 73 74 75 76 77 78 79 80 81 82 83 84 85 86 87 88 89 90 91 92 93 94 95 96 97 98 99 100 101 102 103 104 105 106 107 108 109 110 111 112 113 114 115 116 117 118 119 120 121 122 123 124 125 126 127 128 129 130 131 132 133 134 135 136 137 138 139 140 141 142 143 144 145 146 147 148 149 150 151 152 153 package com.your-app-name;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 com.facebook.react.bridge.Promise;import com.facebook.react.bridge.WritableMap;import com.facebook.react.bridge.Arguments;import com.facebook.react.modules.core.DeviceEventManagerModule;import android.content.Context;import android.app.ActivityManager;import androidx.work.ExistingPeriodicWorkPolicy; import androidx.work.PeriodicWorkRequest;import androidx.work.WorkManager;import java.util.Timer;import java.util.TimerTask;import java.util.List;import javax.annotation.Nullable;import java.util.concurrent.TimeUnit;public class BackgroundPosition extends ReactContextBaseJavaModule { private static ReactApplicationContext reactContext; private Timer timer = null ; private TimerTask task = null ; private PeriodicWorkRequest workRequest; private static final String TAGERROR = "START_BACKGROUND_TASK_ERROR" ; public BackgroundPosition (ReactApplicationContext context) { super (context); reactContext = context; workRequest = new PeriodicWorkRequest .Builder(BackgroundPositionWorker.class, 15 , TimeUnit.MINUTES).build(); } @Override public String getName () { return "BackgroundPosition" ; } private void sendEvent (ReactContext reactContext, String eventName, @Nullable WritableMap params) { reactContext .getJSModule(DeviceEventManagerModule.RCTDeviceEventEmitter.class) .emit(eventName, params); } @ReactMethod public void addListener (String eventName) { } @ReactMethod public void removeListeners (Integer count) { } private boolean isAppOnForeground (Context context) { 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 ; } @ReactMethod public void startBackgroudTask (Promise promise) { if (timer!=null ) { timer.cancel(); timer=null ; } timer = new Timer (); task = new TimerTask () { @Override public void run () { try { if (!isAppOnForeground(reactContext)) { WritableMap params = Arguments.createMap(); params.putString("msg" , "app已经在后台了,准备启动BackgroundPostionWorker" ); sendEvent(reactContext, "backgroundTask" , params); WorkManager.getInstance().enqueueUniquePeriodicWork("BackgroundPositionWorker" , ExistingPeriodicWorkPolicy.KEEP, workRequest); WritableMap params2 = Arguments.createMap(); params2.putString("msg" , "BackgroundPostionWorker started" ); promise.resolve(params2); } } catch (Exception e) { e.printStackTrace(); promise.reject(TAGERROR, e); } } }; timer.schedule(task, 3000 ); } @ReactMethod public void stopBackgroudTask (Promise promise) { if (timer!=null ) { timer.cancel(); timer=null ; } WritableMap params = Arguments.createMap(); params.putString("msg" , "BackgroundPostionWorker stop successed" ); WorkManager.getInstance().cancelUniqueWork("BackgroundPositionWorker" ); promise.resolve(params); } }
android/app/src/main/java/com/your-app-name/BackgroundPositionPackage.java
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 package com.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 BackgroundPositionPackage 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 BackgroundPosition (reactContext)); return modules; } }
android/app/src/main/java/com/your-app-name/MainApplication.java
+ import com.your-app-name.BackgroundPositionPackage; public class MainApplication extends Application implements ReactApplication { ... @Override protected List<ReactPackage> getPackages() { @SuppressWarnings("UnnecessaryLocalVariable") List<ReactPackage> packages = new PackageList(this).getPackages();+ packages.add(new BackgroundPositionPackage());// <-- 添加这一行,类名替换成你的Package类的名字 name. return packages; } ... }
android/app/src/main/java/com/your-app-name/BackgroundPositionServices.java
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 32 33 34 35 36 37 38 39 40 41 42 package com.your-app-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 com.facebook.react.bridge.WritableMap;import com.facebook.react.jstasks.HeadlessJsTaskRetryPolicy;import com.facebook.react.jstasks.LinearCountingRetryPolicy;import javax.annotation.Nullable;public class BackgroundPositionServices extends HeadlessJsTaskService { @Override protected @Nullable HeadlessJsTaskConfig getTaskConfig (Intent intent) { Bundle extras = intent.getExtras(); WritableMap data = extras != null ? Arguments.fromBundle(extras) : Arguments.createMap(); LinearCountingRetryPolicy retryPolicy = new LinearCountingRetryPolicy ( 3 , 1000 ); return new HeadlessJsTaskConfig ( "BackgroundPosition" , data, 10000 , false , retryPolicy ); } }
android/app/src/main/java/com/your-app-name/BackgroundPositionWorker.java
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 package com.your-app-name;import androidx.annotation.NonNull;import androidx.work.Worker; import androidx.work.WorkerParameters;import android.os.Bundle;import android.content.Intent;import android.content.Context;public class BackgroundPositionWorker extends Worker { public BackgroundPositionWorker ( @NonNull Context context, @NonNull WorkerParameters workerParams) { super (context, workerParams); } @NonNull @Override public Result doWork () { Intent service = new Intent (getApplicationContext(), BackgroundPositionServices.class); Bundle bundle = new Bundle (); bundle.putString("msg" , "backgroundPosition start" ); service.putExtras(bundle); getApplicationContext().startService(service); return Result.success(); } }
android/app/src/main/AndroidManifest.xml
中添加权限
... + + <uses-permission android:name ="android.permission.ACCESS_FINE_LOCATION" /> + + <uses-permission android:name ="android.permission.ACCESS_COARSE_LOCATION" /> + + <uses-permission android:name ="android.permission.ACCESS_BACKGROUND_LOCATION" /> <application > ... + <service android:name ="com.your-app-name.BackgroundPositionServices" /> </application >
index.js
中注册后台任务
import {AppRegistry} from 'react-native'; import App from './App'; import {name as appName} from './app.json';+ import {backgroundPosition} from './src/utils'; AppRegistry.registerComponent(appName, () => App);+ AppRegistry.registerHeadlessTask('BackgroundPosition', () => backgroundPosition);
src/utils/backgroundPosition.js
后台任务具体代码
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 32 33 34 35 36 37 38 39 40 41 42 43 44 45 46 47 48 49 50 51 52 53 54 55 56 57 58 59 60 61 62 63 64 65 import {InteractionManager , AppState , NativeModules } from 'react-native' ;import Geolocation from '@react-native-community/geolocation' ;import AsyncStorage from '@react-native-async-storage/async-storage' ;import dayjs from 'dayjs' ;const BackgroundPosition = NativeModules .BackgroundPosition ;function handleListenerAppState (watchId = 0 ) { const subscription = AppState .addEventListener ('change' , nextAppState => { console .log ('nextAppState' , nextAppState); if (nextAppState === 'active' ) { flag = false ; console .log ('app回到前台,后台任务停止' ); console .log ('watchId:' , watchId); BackgroundPosition .stopBackgroudTask (); Geolocation .clearWatch (watchId); subscription.remove (); } }); }export async function backgroundPosition (e ) { await AsyncStorage .clear (); const handle = InteractionManager .createInteractionHandle (); InteractionManager .runAfterInteractions (() => { let watchPositionId = Geolocation .watchPosition ( async info => { const { coords : {latitude, longitude}, } = info; console .log ('当前位置:' , latitude, longitude); let locationListStr = await AsyncStorage .getItem ('location' ); let locationObj = locationListStr === null ? {list : []} : JSON .parse (locationListStr); locationObj.list .push ({ latitude, longitude, date : dayjs ().format ('YYYY-MM-DD HH:mm:ss' ), }); await AsyncStorage .setItem ('location' , JSON .stringify (locationObj)); }, err => { console .warn ('获取定位失败==>' , err); }, { interval : 5000 , timeout : 10000 , maximumAge : 15000 , enableHighAccuracy : true , distanceFilter : 1 , }, ); console .log ('watchPositionId:' , watchPositionId); handleListenerAppState (watchPositionId); }); InteractionManager .clearInteractionHandle (handle); }
页面UI中点击某按钮执行后台任务
Android 10(API 级别 29)中,新增了ACCESS_BACKGROUND_LOCATION后台权限
在android 11级以上版本需要先申请ACCESS_COARSE_LOCATIO和ACCESS_FINE_LOCATION后, 再申请ACCESS_BACKGROUND_LOCATION权限,才能确保前台访问位置权限和后台访问位置权限正常
如果同时申请这三个权限时不会弹窗,系统会忽略权限请求,不会授予其中的任一权限。
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 32 33 34 35 36 37 38 39 40 41 42 43 44 45 46 47 48 49 50 51 52 53 54 55 56 57 const BackgroundPosition = NativeModules .BackgroundPosition ;const handleAndroidPositionPermission = async ( ) => { try { const granted1 = await PermissionsAndroid .requestMultiple ([ PermissionsAndroid .PERMISSIONS .ACCESS_FINE_LOCATION , PermissionsAndroid .PERMISSIONS .ACCESS_COARSE_LOCATION , ]); const granted2 = await PermissionsAndroid .request ( PermissionsAndroid .PERMISSIONS .ACCESS_BACKGROUND_LOCATION , ); if ( granted1['android.permission.ACCESS_FINE_LOCATION' ] === PermissionsAndroid .RESULTS .GRANTED && granted1['android.permission.ACCESS_COARSE_LOCATION' ] === PermissionsAndroid .RESULTS .GRANTED && granted2 === PermissionsAndroid .RESULTS .GRANTED ) { console .log ('可以定位了' ); return Promise .resolve (); } else { console .log ('拒绝获取定位权限' ); Toast .fail ({ content : "拒绝获取定位权限" , duration : 2 , stackable : true , }); return Promise .reject ({msg : '拒绝获取定位权限' }); } } catch (error) { console .warn (error); return Promise .reject (); } };const handleBackgroundTask = async type => { try { if (type === 'start' ) { await handleAndroidPositionPermission (); await BackgroundPosition .startBackgroudTask (); } else { await BackgroundPosition .stopBackgroudTask (); } } catch (error) { console .error ('handleBackgroundTask error' , error); } };
6. 实际测试结果和存在的问题
测试机型小米10,android13
开启后台任务后,手机锁屏,执行20分钟后,app被系统自己杀掉了,如果是在持续玩手机,app没被系统杀掉,可能和手机的省电策略有关;
坐标保存不是很多,甚至出现中途有20分钟没保存坐标,不知道什么原因;
保存的gps坐标,在google地图上和活动轨迹大概吻合,但是误差有点大;
可能是@react-native-async-storage/async-storage
的watchPosition
有问题,需要自定义一个实时获取坐标的安卓原生模块对比测试下
7. 关于在Headless JS中如何执行setTimeout
? ISSUE里搜了下,也没什么关键信息,甚至显示有人已经提交过PR了
那使用requestAnimationFrame
呢?经过实际测试,有时候执行,有时候不执行
setInterval
和setImmediate
也不行
使用while循环,自己实现一个setTimeout,配合递归,就是一个setInterval了,经过实际测试,可行
const sleep = function (startTime, delay ) { return () => { let cur = new Date ().getTime (); while (cur < startTime + delay) { cur = new Date ().getTime (); } }; };function fun ( ) { sleep (new Date ().getTime (), 3000 )(); fun (); }
8. 参考资料
headless-js中文文档
Run React Native background tasks with Headless JS
使用android WorkManager的React Native HeadlessJs任务调用
How to Run a Background Task in React Native ?
android位置权限的变更史