在我关闭应用程序或使用工作管理器,警报管理器或作业调度程序重新启动后,任务永远不会执行



我想实现的目标:

  1. 无论应用程序是否被终止或设备重新启动,该任务都将在后台持续运行。假设我正在将联系人同步到服务器,我需要每分钟检查是否有新联系人,这对我的任务来说是非常重要的要求。
  2. 如果我们转到开发人员设置,然后"运行服务",这些服务永远不会被杀死。并在设备重新启动后不断重启。例如facebookServicesWhatsApp.

我正在使用的物理设备的信息:

Android 10. 
API Level 29

我尝试过的事情:

JobScheduler
WorkManager
Service (startService())
JobIntentService
AlarmManager

我面临的问题:

我已经阅读了很多文章,博客并观看了很多视频,但仍然无法实现我想要的。我也知道如果我使用带有通知通道的前台服务,我的服务不会被杀死并持续存在,但我不希望这样,因为必须有没有任何通知通道的情况下运行 facebookServices 的方式,即使在 API 级别 29 上,它们也只是运行后台服务,在 Android O 之后具有所有后台限制。

  1. 将 AlarmManager 与 BroadcastReceiver 结合使用,后者尝试在特定时间启动服务。但是有连续的异常抛出java.lang.IllegalStateException: Not allowed to start service Intent.
  2. 尝试了workManager,它仅在应用程序打开时而不是在我关闭应用程序时完成工作。如果我重新启动设备,它也不会重新启动工作。
  3. JobScheduler也无法准确获取我想要的内容。

安卓清单.xml

<?xml version="1.0" encoding="utf-8"?>
<manifest xmlns:android="http://schemas.android.com/apk/res/android"
package="com.example.basicplayer">
<uses-permission android:name="android.permission.RECEIVE_BOOT_COMPLETED"/>
<uses-permission android:name="android.permission.WAKE_LOCK"/>
<application
android:allowBackup="true"
android:icon="@mipmap/ic_launcher"
android:label="@string/app_name"
android:roundIcon="@mipmap/ic_launcher_round"
android:supportsRtl="true"
android:theme="@style/AppTheme">
<activity android:name=".MainActivity">
<intent-filter>
<action android:name="android.intent.action.MAIN" />
<category android:name="android.intent.category.LAUNCHER" />
</intent-filter>
</activity>
<service android:name=".MyAlarmService"
android:enabled="true"
android:exported="true"
/>
<receiver android:name=".MyAlarmReceiver"
android:enabled="true"
android:exported="true"
>
<intent-filter>
<action android:name="android.intent.action.BOOT_COMPLETED" />
</intent-filter>
</receiver>
</application>
</manifest>

我的警报接收器.java

package com.example.basicplayer;
import android.content.BroadcastReceiver;
import android.content.ComponentName;
import android.content.Context;
import android.content.Intent;
import android.media.MediaPlayer;
import android.os.Build;
import android.provider.Settings;
import androidx.annotation.RequiresApi;
public class MyAlarmReceiver extends BroadcastReceiver {

@RequiresApi(api = Build.VERSION_CODES.O)
@Override
public void onReceive(Context context, Intent intent) {
context.startService(new Intent(context, MyAlarmService.class));
}
}

我的警报服务.java

package com.example.basicplayer;
import android.app.Notification;
import android.app.Service;
import android.content.Intent;
import android.media.MediaPlayer;
import android.os.IBinder;
import android.provider.Settings;
import androidx.annotation.Nullable;
public class MyAlarmService extends Service {
private MediaPlayer mediaPlayer;
@Override
public int onStartCommand(Intent intent, int flags, int startId) {
mediaPlayer = MediaPlayer.create(this,
Settings.System.DEFAULT_ALARM_ALERT_URI);
mediaPlayer.setLooping(true);
mediaPlayer.start();
return START_STICKY;
}
@Override
public void onCreate() {
super.onCreate();
}
@Override
public void onDestroy() {
super.onDestroy();
try{
startService(new Intent(this, MyAlarmService.class));
}catch (Exception e){
e.printStackTrace();
}

}
@Nullable
@Override
public IBinder onBind(Intent intent) {
return null;
}
}

主要活动.java

package com.example.basicplayer;
import androidx.appcompat.app.AppCompatActivity;
import android.app.AlarmManager;
import android.app.PendingIntent;
import android.content.Context;
import android.content.Intent;
import android.media.MediaPlayer;
import android.os.Bundle;
import android.provider.Settings;
import android.widget.Toast;
public class MainActivity extends AppCompatActivity {
@Override
protected void onCreate(Bundle savedInstanceState) {
super.onCreate(savedInstanceState);
setContentView(R.layout.activity_main);
setAlarm();
}

private void setAlarm() {
//getting the alarm manager
AlarmManager am = (AlarmManager) getSystemService(Context.ALARM_SERVICE);
//creating a new intent specifying the broadcast receiver
Intent i = new Intent(this, MyAlarmReceiver.class);
//creating a pending intent using the intent
PendingIntent pi = PendingIntent.getBroadcast(this, 0, i, 0);
//setting the repeating alarm that will be fired every day
am.setRepeating(AlarmManager.RTC, 10000, AlarmManager.INTERVAL_FIFTEEN_MINUTES, pi);
Toast.makeText(this, "Alarm is set", Toast.LENGTH_SHORT).show();
}
}

你们能帮忙吗?

在新版安卓中,对后台运行进程引入了限制。 他们这样做的原因是为了提供电池优化。我认为你应该使用工作管理器库。它是用于后台进程的真棒库。此外,当设备重新启动时,您可以使用工作管理器启动作业。我为您找到了一个答案:在安卓奥利奥中重新启动设备后,定期工作请求不起作用

最新更新