web-dev-qa-db-ja.com

AlarmManagerが間違ったタイミングでアラームを起動する

私はすべてを管理して、アラームの結果として通知を起動するために使用される通知サービスを作成しました。残念ながら、AlarmManagerを使用したアラームの設定は正しく機能しません。数分後に起動します(正確には数時間ではなく、タイムゾーンの問題を示します)。繰り返しの期間は1週間なので、定数INTERVAL_DAYを使用して7を掛けました。一方のPendingIntentが他方を置き換えないようにするために、dayOfWeekを2番目のパラメーターとしてPendingIntent.getService()に渡します。ログに記録して、アラームが発生する時間の正確さを確認します。

Log.d(TAG, "next alarm " + df.format(cal.getTime()));

設定されたすべてのアラームを一覧表示する方法は本当にありませんか?少なくとも自分のアプリからのものは?これがエラーを追跡する唯一の方法だと思います。

私のコード:

cal.setTimeInMillis(System.currentTimeMillis());
cal.add(Calendar.DATE, 1);
cal.set(Calendar.HOUR_OF_DAY, hour);
cal.set(Calendar.MINUTE, minute);
Log.d(TAG, "next alarm " + df.format(cal.getTime()));
Intent showNotificationIntent = new Intent(context, NotificationService.class);
dayOfWeek = cal.get(Calendar.DAY_OF_WEEK);
alarmIntent = PendingIntent.getService(context, dayOfWeek, showNotificationIntent, 0);
getAlarmManager(context).setInexactRepeating(AlarmManager.RTC_WAKEUP, cal.getTimeInMillis(),
    INTERVAL_WEEK, alarmIntent);

私は毎日アラームを提供したいと思いますが、ユーザーが設定できるさまざまなタイミングで。したがって、私は最大7つのアラームを使用しており、毎週発生するはずです。

同様の質問に対する多数の回答を読んでも(重複する質問を作成するつもりはありません)、問題を見つけることができませんでした。

17
Holger Jakobs

19未満のAPIレベルの場合、AlarmManager.setRepeating()を使用する必要があり、アラームは指定された時間に正確にトリガーされます。 apiレベル19以上では、これは機能しなくなります。 Android=に変更があったため、すべての繰り返しアラームが不正確です。そのため、正確な繰り返しアラームを達成したい場合は、AlarmManager.setExact()を使用してアラームをスケジュールしてから、アラームトリガーが来週もまた毎週同じように。

26
Blaz

SetInexactRepeatingのため。 setRepeatingを使用すると、適切なタイミングで処理されます。

の代わりに:

setInexactRepeating 

使用する

setRepeating

setInexactRepeatingはOSとバッテリーに優しいもので、アラーム受信で行われるすべての作業をバッチ処理し、1つずつ処理しながら、setRepeatingが即座にアラームを起動します

また、注意事項:電話を再起動するとアラームは消去されます。永続化するには、ブートブロードキャストレシーバーを実装する必要がある場合があります。そのランタイムを実行しないようにしてください。アプリがバックグラウンドにない場合は、マニフェストにそれを実装する必要があります。ブロードキャストは受信されません。

小さな例:

これは実際のコードです。電話がオフになるまで、10分ごとにCPUをウェイクアップします。

Manifest.xmlに追加:

...
<uses-permission Android:name="Android.permission.WAKE_LOCK"></uses-permission>
...
<receiver  Android:process=":remote" Android:name="Alarm"></receiver>
...

コード:

    package YourPackage;
    import Android.app.AlarmManager;
    import Android.app.PendingIntent;
    import Android.content.BroadcastReceiver;
    import Android.content.Context;
    import Android.content.Intent;
    import Android.os.PowerManager;
    import Android.widget.Toast;

    public class Alarm extends BroadcastReceiver 
    {    
         @Override
         public void onReceive(Context context, Intent intent) 
         {   
             PowerManager pm = (PowerManager) context.getSystemService(Context.POWER_SERVICE);
             PowerManager.WakeLock wl = pm.newWakeLock(PowerManager.PARTIAL_WAKE_LOCK, "");
             wl.acquire();

             // Put here YOUR code.
             Toast.makeText(context, "Alarm !!!!!!!!!!", Toast.LENGTH_LONG).show(); // For example

             wl.release();
         }

     public void SetAlarm(Context context)
     {
         AlarmManager am=(AlarmManager)context.getSystemService(Context.ALARM_SERVICE);
         Intent i = new Intent(context, Alarm.class);
         PendingIntent pi = PendingIntent.getBroadcast(context, 0, i, 0);
         am.setRepeating(AlarmManager.RTC_WAKEUP, System.currentTimeMillis(), 1000 * 60 * 10, pi); // Millisec * Second * Minute
     }

     public void CancelAlarm(Context context)
     {
         Intent intent = new Intent(context, Alarm.class);
         PendingIntent sender = PendingIntent.getBroadcast(context, 0, intent, 0);
         AlarmManager alarmManager = (AlarmManager) context.getSystemService(Context.ALARM_SERVICE);
         alarmManager.cancel(sender);
     }
 }

サービスからアラームを設定:

package YourPackage;
import Android.app.Service;
import Android.content.Context;
import Android.content.Intent;
import Android.os.IBinder;

public class YourService extends Service
{
    Alarm alarm = new Alarm();
    public void onCreate()
    {
        super.onCreate();       
    }

    public void onStart(Context context,Intent intent, int startId)
    {
        alarm.SetAlarm(context);
    }

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

電話の起動時に繰り返しアラームを設定する場合:

Manifest.xmlに権限を追加します。

<uses-permission Android:name="Android.permission.RECEIVE_BOOT_COMPLETED"></uses-permission>
...
<receiver Android:name=".AutoStart">
    <intent-filter>
        <action Android:name="Android.intent.action.BOOT_COMPLETED"></action>
    </intent-filter>
</receiver>
...

そして、新しいクラスを作成します。

package YourPackage;
import Android.content.BroadcastReceiver;
import Android.content.Context;
import Android.content.Intent;

public class AutoStart extends BroadcastReceiver
{   
    Alarm alarm = new Alarm();
    @Override
    public void onReceive(Context context, Intent intent)
    {   
        if (intent.getAction().equals("Android.intent.action.BOOT_COMPLETED"))
        {
            alarm.SetAlarm(context);
        }
    }
}
13
Skynet