5

目標:特定の条件が満たされた場合、通知は毎日午後 2 時に 1 回表示されます。

例:簡単にするために、インターネット接続でチェックされる条件が毎日満たされていると考えてみましょう。今日が午後 2 時を過ぎている場合は、明日から通知を開始します。たとえば、ユーザーが月曜日の午後 4 時にアプリを起動すると、火曜日の午後 2 時、水曜日の午後 2 時、木曜日の午後 2 時などに通知を受け取ります。

問題:午後 2 時に最初の通知がありますが、ランダムな時間に同じ通知を何度も受け取ります。

問題は Android >= 4.0 でのみ発生するようです。以前の Android では問題なく動作します。

これは私が通知を送信する方法です:

public class NotifyService extends Service
{       
static final int NOTIFICATION_ID = 1;
// ...

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

@Override
public int onStartCommand(Intent intent, int flags, int startId)
{
    try
    {
        Symbol biggest = getBiggestMover();
        if (biggest != null)
        {
            String title = getString(R.string.app_name);
            String text = getNotificationText(biggest.symbol, biggest.change);
            sendNotification(title, text);
        }
    }
    catch (Exception e)
    {
        // If there is Internet problem we do nothing, don't want to disturb the user.
        e.printStackTrace();
    }

    return super.onStartCommand(intent, flags, startId);
}

/** @return Symbol which is the biggest mover today. If there is no big mover - null is returned.
 * @throws Exception If there is Internet problem. */
private Symbol getBiggestMover() throws Exception
{
    Symbol biggest = null;
    Symbol[] equities = Network.getTraded(SymbolType.EQUITY);
    for (Symbol equity : equities)
    {
        if (Utilities.isToday(equity.lastTraded) && isBigMove(equity.change) && isBigger(equity, biggest))
        {
            biggest = equity;
        }
    }
    return biggest;
}   

private void sendNotification(String title, String text)
{
    Notification notification = new Notification(R.drawable.ic_launcher, text, System.currentTimeMillis());
    notification.flags = Notification.FLAG_AUTO_CANCEL;

    Intent clickIntent = new Intent(this, MainActivity.class);
    PendingIntent pendingIntent = PendingIntent.getActivity(this, 0, clickIntent, PendingIntent.FLAG_CANCEL_CURRENT);

    notification.setLatestEventInfo(this, title, text, pendingIntent);

    NotificationManager manager = (NotificationManager) getSystemService(Context.NOTIFICATION_SERVICE);
    manager.notify(NOTIFICATION_ID, notification);
}   
// ...  
}

sendNotification()は次の理由により、午後 2 時に呼び出されAlarmManagerます。

public class ServiceStarter extends BroadcastReceiver
{

@Override
public void onReceive(Context context, Intent intent)
{
    setNotificationAlarm(context);
}

/** Set repeating notifications every 24 hours. */
public static void setNotificationAlarm(Context context)
{
    Intent intent = new Intent(context, NotifyService.class);
    AlarmManager alarmManager = (AlarmManager) context.getSystemService(Context.ALARM_SERVICE);
    PendingIntent pendingIntent = PendingIntent.getService(context, 0, intent, PendingIntent.FLAG_CANCEL_CURRENT);

    final int oneDay = 24 * 60 * 60 * 1000;
    alarmManager.setRepeating(AlarmManager.RTC, getTriggerTime(), oneDay, pendingIntent);
}

private static long getTriggerTime()
{
    GregorianCalendar calendar = new GregorianCalendar();
    calendar.set(GregorianCalendar.HOUR_OF_DAY, 14);
    calendar.set(GregorianCalendar.MINUTE, 0);
    calendar.set(GregorianCalendar.SECOND, 0);
    calendar.set(GregorianCalendar.MILLISECOND, 0);

    if (calendar.before(new GregorianCalendar()))
    {
        calendar.add(GregorianCalendar.DAY_OF_MONTH, 1);
    }

    return calendar.getTimeInMillis();
}

}

setNotificationAlarm()2箇所から呼び出されます。まずはアプリ起動時。次に、上記のコードから、電話が再起動したとき ( onReceive()receive BOOT_COMPLETED)。ユーザーが電話をオフにするとAlarmManager、アラームがクリアされるためです。

alarmManager.setRepeating()以前のアラームを上書きするため、すべてが機能するはずです。

誰かが同じ問題を抱えていることを発見しましたが、回答もありません:
https://groups.google.com/forum/?fromgroups=#!topic/android-developers/t_tDU4PwR3g

ここでも同様の問題が見つかりました: http://comments.gmane.org/gmane.comp.handhelds.android.devel/171471

少し前に、そのような通知を作成する方法を尋ねたので、これは関連しています:
特定の時間の毎日の通知

4

2 に答える 2

1

ICS+ デバイスでも同じ問題が発生しました。私の回避策は非常に簡単でした->通知が表示されたら、現在の時間を共有設定に入れます。その前に、間隔が本当に経過したかどうかを常に確認し、そうでない場合は中止します。

            long lastnotification = sharedPrefs.getLong("lnnd", -1);
            Calendar now = Calendar.getInstance();
            if (!namedayalarmEnabled) {
                    return;
            }
            if (lastnotification > 1) {
                    Calendar last = Calendar.getInstance();
                    last.setTimeInMillis(lastnotification);
                    long distance = (now.getTimeInMillis() - last
                                    .getTimeInMillis());
                    if (distance < YOURINTERVAL) {
                            return;
                    } else {
                            SharedPreferences.Editor editor = sharedPrefs.edit();
                            editor.putLong("lnnd", now.getTimeInMillis());
                            editor.commit();
                    }
            }
于 2012-11-01T13:40:46.480 に答える
1

AlarmManager.RTC の代わりに AlarmManager.RTC_WAKEUP を使用します。

AlarmManager.RTC 内

System.currentTimeMillis() のアラーム時間 (UTC の壁時計時間)。このアラームはデバイスを起動しません。デバイスがスリープ中にオフになると、次にデバイスがウェイクアップするまで配信されません。

AlarmManager.RTC_WAKEUP の場合

System.currentTimeMillis() のアラーム時間 (UTC の壁時計時間)。オフになったときにデバイスをウェイクアップします。

于 2012-10-30T07:22:59.563 に答える