android 7 JobScheduler实现APP保活

前言

JobScheduler:当一系列预置的条件被满足时,JobScheduler API为你的应用执行一个操作,例如当设备接通电源适配器或者连接到WIFI,在API 21 ( Android 5.0(Lollipop) )中,google提供了一个新叫做JobScheduler API的组件来处理这样的场景。
在API 24 ( Android 7.0(N) )的新特性中,Google对于新设备功耗要求越来越严格,对于APP的限制也越来约多,想继续像Android5.0、6.0一样简单处理定时执行执行周期性作业,需要做出一些代码调整,针对该需求,总结了2种解决方案供参考。

  • setPeriodic
    setPeriodic:按时间间隔执行周期性作业,在Android 5、6平台版本下可以间隔任何时间运行,在Android7.0平台版本上需设置定期作业的间隔时间>=15分钟时才能运行。

  • setMinimumLatency
    设置作业延迟执行的时间,与setPeriodic不可同时执行,可配置setOverrideDeadline设置作业最大延迟执行时间。

  • setRequiredNetworkType
    设置作业只有在满足指定的网络条件时才会被执行

/** 默认条件,不管是否有网络这个作业都会被执行 */
public static final int NETWORK_TYPE_NONE = 0;
/** 任意一种网络这个作业都会被执行 */
public static final int NETWORK_TYPE_ANY = 1;
/** 不是蜂窝网络( 比如在WIFI连接时 )时作业才会被执行 */
public static final int NETWORK_TYPE_UNMETERED = 2;
/** 不在漫游时作业才会被执行 */
public static final int NETWORK_TYPE_NOT_ROAMING = 3;

解决方案1

使用Context.getSystemService(Context.JOB_SCHEDULER_SERVICE)创建JobScheduler对象,配置调度工作setMinimumLatency等参数,延迟运行Job Service,在JobSchedulerService服务的onStartJob方法中启动保活服务,再创建一个新的JobScheduler任务,并结束当前JobScheduler任务。

如果正在运行需要很长时间的任务,则下一个任务将安排在当前任务完成时间+ PROVIDED_TIME_INTERVAL

  • 创建JobScheduler
    设置setExtras参数,将需要保活的服务名称传递至Job Service
 //API大于24
 if (Build.VERSION.SDK_INT >= Build.VERSION_CODES.N) {
                    //7.0+
                    mJobScheduler = (JobScheduler) mContext.getSystemService(Context.JOB_SCHEDULER_SERVICE);
                    JobInfo.Builder builder = new JobInfo.Builder(1,
                            new ComponentName(mContext.getPackageName(), JobSchedulerService.class.getName()));
                    builder.setMinimumLatency(2 * 60 * 1000);
                    PersistableBundle persiBundle = new PersistableBundle();
                    persiBundle.putString("servicename", service.getName());
                    builder.setExtras(persiBundle);
                    if (mJobScheduler.schedule(builder.build()) <= 0) {
                        //If something goes wrong
                    }
                }
  • 执行JobSchedulerService 工作任务
    在onStartJob方法中,执行startService方法启动保活服务,创建一个新的JobScheduler对象,手动调用jobFinished(params, false)结束作业,并将返回值设为true
onStartJob的返回值区别:
(1) false:框架认为你作业已经执行完毕了,那么下一个作业就立刻展开了
(2) true:框架将作业结束状态交给你去处理。因为我们可能会异步的通过线程等方式去执行工作,这个时间肯定不能放在主线程里面去控制,这时候需要手动调用jobFinished(JobParameters params, boolean needsReschedule)方法去告诉框架作业结束了,其中needsReschedule表示是否重复执行
public boolean onStartJob(JobParameters params) {
        try {
            Log.d("JobSchedulerService","start~~"+ System.currentTimeMillis());
            if (Build.VERSION.SDK_INT >= Build.VERSION_CODES.N) {
                Log.d("JobSchedulerService", "7.0 handleMessage task running");
                String servicename = params.getExtras().getString("servicename");
                Class service = getClassLoader().loadClass(servicename);
                if (service != null) {
                    Log.d("JobSchedulerService", "7.0 handleMessage task running ~~2~~"+service.hashCode());
                    //判断保活的service是否被杀死
                    if (!isMyServiceRunning(service)) {
                        //重启service
                        startService(new Intent(getApplicationContext(), service));
                    }
                }
                //创建一个新的JobScheduler任务
                scheduleRefresh(servicename);
                jobFinished(params, false);
                Log.d("JobSchedulerService","7.0 handleMessage task end~~"+ System.currentTimeMillis());
                return true;
            }
        } catch (Exception e) {
            e.printStackTrace();
        }
        return false;
    }
  • scheduleRefresh
private void scheduleRefresh(String serviceName) {
        JobScheduler mJobScheduler = (JobScheduler)getApplicationContext()
                .getSystemService(JOB_SCHEDULER_SERVICE);
        //jobId可根据实际情况设定        
        JobInfo.Builder mJobBuilder =
                new JobInfo.Builder(0,
                        new ComponentName(getPackageName(),
                                JobSchedulerService.class.getName()));

        if (android.os.Build.VERSION.SDK_INT >= Build.VERSION_CODES.N) {
            mJobBuilder.setMinimumLatency(2* 60 * 1000).setRequiredNetworkType(JobInfo.NETWORK_TYPE_ANY);
            PersistableBundle persiBundle = new PersistableBundle();
            persiBundle.putString("servicename", serviceName);
            mJobBuilder.setExtras(persiBundle);
        }

        if (mJobScheduler != null && mJobScheduler.schedule(mJobBuilder.build())
                <= JobScheduler.RESULT_FAILURE) {
            //Scheduled Failed/LOG or run fail safe measures
            Log.d("JobSchedulerService", "7.0 Unable to schedule the service FAILURE!");
        }else{
            Log.d("JobSchedulerService", "7.0 schedule the service SUCCESS!");
        }
    }
  • 判断保活Service是否在运行
    public boolean isMyServiceRunning(Class serviceClass) {
        ActivityManager manager = (ActivityManager) getSystemService(Context.ACTIVITY_SERVICE);
        for (ActivityManager.RunningServiceInfo service : manager.getRunningServices(Integer.MAX_VALUE)) {
            if (serviceClass.getName().equals(service.service.getClassName())) {
                return true;
            }
        }
        return false;
    }

解决方案(作业间隔时间>=15分钟)

设定作业的间隔时间大于15分钟,采用setPeriodic方法实现定时执行周期性作业。

private static final long REFRESH_INTERVAL = 15 * 60 * 1000;

  • Job Scheduler代码
public static void scheduleJob(Context context) {
    ComponentName serviceComponent = new ComponentName(context, PAJobService.class);
    JobInfo.Builder builder = new JobInfo.Builder(JOB_ID, serviceComponent);
    builder.setPeriodic(15 * 60 * 1000, 5 * 60 *1000);

    JobScheduler jobScheduler = context.getSystemService(JobScheduler.class);
    int ret = jobScheduler.schedule(builder.build());
    if (ret == JobScheduler.RESULT_SUCCESS) {
        Log.d(TAG, "Job scheduled successfully");
    } else {
        Log.d(TAG, "Job scheduling failed");
    }
}
  • JobService服务
public class PAJobService extends JobService {
    private static final String TAG = PRE_TAG + PAJobService.class.getSimpleName();
    private LocationManager mLocationManager;

    public boolean onStartJob(JobParameters params) {
        Log.d(TAG, "onStartJob");
        Toast.makeText(getApplicationContext(), "Job Started", Toast.LENGTH_SHORT).show();
        return false;
    }

    public boolean onStopJob(JobParameters params) {
        Log.d(TAG, "onStopJob");
        return false;
    }
}

参考资料:
https://developer.android.google.cn/reference/android/app/job/JobScheduler.html
2、http://blog.csdn.net/bboyfeiyu/article/details/44809395
3、http://blog.csdn.net/aroundme/article/details/55214203

你可能感兴趣的:(Android)