我管理好一切,可以创建一个通知服务,用于通过警报触发通知。不幸的是,使用AlarmManager设置闹钟无法正常工作。它在几分钟后发射(不完全是几小时,这表明存在时区问题)。重复周期是1周,所以我使用常量INTERVAL_DAY并将其乘以7.为了确保一个PendingIntent不替换另一个,我将dayOfWeek作为第二个参数传递给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个警报,每周应该发射一次。
即使在阅读了类似问题的大量答案后(我不打算创建重复的问题),我也没有设法找到问题。
答案 0 :(得分:25)
对于低于19的api级别,您应该使用AlarmManager.setRepeating()
,并且您的警报将在指定时间准确触发。在api级别19及以上,这将不再有效。 android中有变化,所有重复警报都是不准确的。因此,如果您想要实现精确的重复警报,您应该使用AlarmManager.setExact()
安排警报,然后在下周警报触发时再进行警报,依此类推。
答案 1 :(得分:14)
因为setInexactRepeating。使用setRepeating,它将在合适的时间处理。
而不是:
setInexactRepeating
使用
setRepeating
setInexactRepeating,是操作系统和电池友好的,它将所有要完成的工作批量接收到报警接收并逐个工作,同时setRepeating立即触发报警
另请注意:重新启动电话后会擦除警报,您可能必须实施启动广播接收器以使其持久。确保你没有那个运行时,你需要在Manifest中实现它,当你的应用程序不在后台你不会收到任何广播。
这是有效的代码。它会每隔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);
}
}
}