使用API​​ 19+以较短的间隔更新Widget

Orb*_*bit 6 android android-widget android-handler

Pre API 19,比updatePeriodMillis最小时间30分钟更新Widget 的首选方法是在设置AlarmManager时使用指定的间隔后使用a AlarmManager和a BroadcastReceiver接收Intent.

目前,使用下面的代码,Widget会更新,但是从Android 5.1开始,使用重复间隔小于60000ms的.setRepeating()会自动将其间隔设置为至少60000ms.

在Widgets onEnabled()中设置闹钟:

AlarmManager am= (AlarmManager)context.getSystemService(Context.ALARM_SERVICE);
Intent intent = new Intent(context, AlarmManagerBroadcastReceiver.class);
PendingIntent pi = PendingIntent.getBroadcast(context, 0, intent, 0);
//After after 3 seconds
am.setRepeating(AlarmManager.RTC_WAKEUP, System.currentTimeMillis()+ 3000, 1000 , pi);
Run Code Online (Sandbox Code Playgroud)

然后在AlarmManagerBroadcastReceiver的onReceive()中:

PowerManager pm = (PowerManager) context.getSystemService(Context.POWER_SERVICE);
PowerManager.WakeLock wl = pm.newWakeLock(PowerManager.PARTIAL_WAKE_LOCK, "TAG");
//Acquire the lock
wl.acquire();

/* 
 * ......
 * Update Widgets RemoteViews
 */

wl.release();
Run Code Online (Sandbox Code Playgroud)

setRepeating()的文档中,它说:

注意:从API 19开始,所有重复警报都不准确.如果您的应用程序需要精确的交付时间,那么它必须使用一次性精确警报,每次重新安排如上所述.targetSdkVersion早于API 19的旧应用程序将继续将所有警报(包括重复警报)视为精确警报.

它现在还说:

安排重复警报.注意:对于计时操作(刻度,超时等),使用起来更容易,效率更高Handler

那么您将如何使用处理程序更新Widgets Remoteviews?当设备进入睡眠状态以节省电池时,您会如何停止它?

有没有其他建议的方法来更新Widget?

cse*_*nga 4

从 API 级别 21 开始,建议使用JobScheduler来处理此类定期更新。

在JobService中定义作业:

public class UpdateJob extends JobService {
    
    public static int JOB_ID=9;
    
    @Override
    public boolean onStartJob(JobParameters params) {
        Toast.makeText(getApplicationContext(),"update",Toast.LENGTH_SHORT).show();
        //call handler, create thread, asynctask etc 
        return false;
    }

    @Override
    public boolean onStopJob(JobParameters params) {
        return false;
    }
}
Run Code Online (Sandbox Code Playgroud)

在清单中注册它:

   <service android:name=".UpdateJob"
            android:permission="android.permission.BIND_JOB_SERVICE" />
Run Code Online (Sandbox Code Playgroud)

安排作业,例如在活动中:

 JobScheduler mJobScheduler = (JobScheduler)
        getSystemService(Context.JOB_SCHEDULER_SERVICE);

JobInfo.Builder builder = new JobInfo.Builder( UpdateJob.JOB_ID,
new ComponentName( getApplicationContext(), UpdateJob.class )  );

builder.setPeriodic(60 * 60 * 1000); // every hour
builder.setRequiredNetworkType(JobInfo.NETWORK_TYPE_ANY); // only when network is available

if( mJobScheduler.schedule( builder.build() ) <= 0 )
{
  //error, cant be scheduled
}

//later e.g. when update is disabled
mJobScheduler.cancel(UpdateJob.JOB_ID);
Run Code Online (Sandbox Code Playgroud)

JobInfo.Builder有很多选项可以在您的作业触发时进行自定义,例如它可以取决于网络和设备状态。

在较低的 API 上,JobSchedulerCompatGCM Network Manager可以用作替代方案,它们的工作方式与上面所示的方式几乎相同。

+还有一个

为了处理获取唤醒锁的 BroadcastReceiver,支持库中有一个“帮助器”类WakefulBroadcastReceiver