博客
关于我
强烈建议你试试无所不能的chatGPT,快点击我
安卓电量优化之JobScheduler使用介绍
阅读量:4488 次
发布时间:2019-06-08

本文共 14852 字,大约阅读时间需要 49 分钟。

版权声明:本文出自汪磊的博客,转载请务必注明出处。

一、JobScheduler概述

JobScheduler是安卓5.0版本推出的API,允许开发者在符合某些条件时创建执行在后台的任务。在Android开发中,会存在这些场景:你需要在稍后的某个时间点或者当满足某个特定的条件时执行一个任务,例如当设备接通电源适配器或者连接到WIFI,此时就可以使用JobScheduler了,当一系列预置的条件被满足时,JobScheduler API为你的应用执行一个操作。与AlarmManager不同的是这个执行时间是不确定的。除此之外,JobScheduler API允许同时执行多个任务。

JobSchedule的宗旨就是把一些不是特别紧急的任务放到更合适的时机批量处理。这样做有两个好处:避免频繁的唤醒硬件模块,造成不必要的电量消耗以及避免在不合适的时间(例如低电量情况下、弱网络或者移动网络情况下的)执行过多的任务消耗电量。

JobSchedule适用版本为5.0及以上,目前还未发现兼容库支持。

二、JobScheduler使用

使用JobScheduler首先我们需要创建一个类继承自JobService且必须实现两个方法(JobService继承自Service),分别是onStartJob(JobParameters params)onStopJob(JobParameters params),如下:

1 public class TestJobService extends JobService { 2  3     @Override 4     public boolean onStartJob(JobParameters params) { 5  6         return false; 7     } 8  9     @Override10     public boolean onStopJob(JobParameters params) {11 12         return false;13     }14 }

当任务开始时会执行onStartJob(JobParameters params)方法,这是系统用来触发已经被执行的任务。这个方法返回一个boolean值。

如果返回值是false,系统假设这个方法返回时任务已经执行完毕。

如果返回值是true,那么系统假定这个任务正要被执行,执行任务的重担就落在了开发者的肩上,当任务执行完毕时开发者需要自己调用jobFinished(JobParameters params, boolean needsRescheduled)来通知系统。

当系统接收到一个取消请求时,会调用onStopJob(JobParameters params)方法取消正在等待执行的任务。很重要的一点是如果onStartJob(JobParameters params)返回false,那么系统假定在接收到一个取消请求时已经没有正在运行的任务。onStopJob(JobParameters params)在这种情况下不会被调用。

需要注意的是这个job service运行在你的主线程,这意味着你需要使用子线程,handler, 或者一个异步任务来运行耗时的操作以防止阻塞主线程。我们可以在onStartJob中利用Handler发送消息,在Handler中处理相关任务。

jobFinished(JobParameters params, boolean needsRescheduled)的两个参数中的params参数是从JobService的onStartJob(JobParameters params)的params传递过来的,needsRescheduled参数是告诉系统这个任务如果由于某些原因导致执行失败是否需要重新调度执行,true需要重新调度执行,false不需要。

最后我们需要到AndroidManifest.xml中添加一个service节点让你的应用拥有绑定和使用这个JobService的权限。

1 
4

在创建完TestJobService后,我们就该创建JobScheduler对象了,创建JobScheduler对象需要通过getSystemService( Context.JOB_SCHEDULER_SERVICE )来初始化:

1 JobScheduler tm =(JobScheduler) getSystemService(Context.JOB_SCHEDULER_SERVICE);

接下来我们使用JobInfo.Builder来构建一个JobInfo对象。JobInfo.Builder接收两个参数,第一个参数是你要运行的任务的标识符,简单理解为这个任务的ID就可以了,第二个是这个Service组件的类名。

1 JobInfo.Builder builder = new JobInfo.Builder(kJobId++, mServiceComponent);

mServiceComponent初始化如下:

1 mServiceComponent = new ComponentName(this, TestJobService.class);

builder允许你设置很多不同的选项来控制任务的执行,重要方法说明如下:

方法名 说明
setPeriodic(long intervalMillis) 设置任务每隔intervalMillis运行一次
setMinimumLatency(long minLatencyMillis) 这个函数能让你设置任务的延迟执行时间(单位是毫秒),这个函数与setPeriodic(long time)方法不兼容,如果这两个方法同时调用了就会引起异常
setOverrideDeadline(long maxExecutionDelayMillis) 这个方法让你可以设置任务最晚的延迟时间。如果到了规定的时间时其他条件还未满足,你的任务也会被启动。与setMinimumLatency(long time)一样,这个方法也会与setPeriodic(long time),同时调用这两个方法会引发异常
setPersisted(boolean isPersisted) 这个方法告诉系统当你的设备重启之后你的任务是否还要继续执行
setRequiredNetworkType(int networkType) 这个方法让你这个任务只有在满足指定的网络条件时才会被执行。默认条件是JobInfo.NETWORK_TYPE_NONE,这意味着不管是否有网络这个任务都会被执行。另外两个可选类型,一种是JobInfo.NETWORK_TYPE_ANY,它表明需要任意一种网络才使得任务可以执行。另一种是JobInfo.NETWORK_TYPE_UNMETERED,它表示设备不是蜂窝网络( 比如在WIFI连接时 )时任务才会被执行
setRequiresCharging(boolean requiresCharging) 这个方法告诉你的应用,只有当设备在充电时这个任务才会被执行
setRequiresDeviceIdle(boolean requiresDeviceIdle) 这个方法告诉你的任务只有当用户没有在使用该设备且有一段时间没有使用时才会启动该任务

需要注意的是setRequiredNetworkType(int networkType), setRequiresCharging(boolean requireCharging) and setRequiresDeviceIdle(boolean requireIdle)者几个方法可能会使得你的任务无法执行,除非调用setOverrideDeadline(long time)设置了最大延迟时间,使得你的任务在未满足条件的情况下也会被执行。一旦你预置的条件被设置,你就可以构建一个JobInfo对象,然后通过如下所示的代码将它发送到你的JobScheduler中

1 if( tm.schedule( builder.build() ) <= 0 ) {2     //something wrong3 }

schedule方法会返回一个整型。如果schedule方法失败了,它会返回一个小于0的错误码。否则返回我们在JobInfo.Builder中定义的标识id。

停止某个任务,可以调用JobScheduler对象的cancel(int jobId)来实现;如果想取消所有的任务,可以调用JobScheduler对象的cancelAll()来实现。

以上就是JobScheduler的介绍,是不是很简单,重要的是理解什么情况下使用JobScheduler

三、JobScheduler实例举例

本Demo是安卓官方提供的一个例子,我们先看TestJobService类;

1 public class TestJobService extends JobService { 2     private static final String TAG = "SyncService"; 3  4     @Override 5     public void onCreate() { 6         super.onCreate(); 7         Log.i(TAG, "Service created"); 8     } 9 10     @Override11     public void onDestroy() {12         super.onDestroy();13         Log.i(TAG, "Service destroyed");14     }15 16 17     @Override18     public int onStartCommand(Intent intent, int flags, int startId) {19         Messenger callback = intent.getParcelableExtra("messenger");20         Message m = Message.obtain();21         m.what = MainActivity.MSG_SERVICE_OBJ;22         m.obj = this;23         try {24             callback.send(m);25         } catch (RemoteException e) {26             Log.e(TAG, "Error passing service object back to activity.");27         }28         return START_NOT_STICKY;29     }30 31     @Override32     public boolean onStartJob(JobParameters params) {33         // We don't do any real 'work' in this sample app. All we'll34         // do is track which jobs have landed on our service, and35         // update the UI accordingly.36        jobParamsMap.add(params);37         if (mActivity != null) {38             mActivity.onReceivedStartJob(params);39         }40         Log.i(TAG, "on start job: " + params.getJobId());41         return true;42     }43 44     @Override45     public boolean onStopJob(JobParameters params) {46         // Stop tracking these job parameters, as we've 'finished' executing.47         jobParamsMap.remove(params);48         if (mActivity != null) {49             mActivity.onReceivedStopJob();50         }51         Log.i(TAG, "on stop job: " + params.getJobId());52         return true;53     }54 55    MainActivity mActivity;56     private final LinkedList
jobParamsMap = new LinkedList
();57 58 public void setUiCallback(MainActivity activity) {59 mActivity = activity;60 }61 62 /** Send job to the JobScheduler. */63 public void scheduleJob(JobInfo t) {64 Log.d(TAG, "Scheduling job");65 JobScheduler tm =66 (JobScheduler) getSystemService(Context.JOB_SCHEDULER_SERVICE);67 if(tm.schedule(t)<=0){68 Log.i(TAG, "something wrong");69 }70 }71 72 73 /**74 * Not currently used, but as an exercise you can hook this75 * up to a button in the UI to finish a job that has landed76 * in onStartJob().77 */78 public boolean callJobFinished() {79 JobParameters params = jobParamsMap.poll();80 if (params == null) {81 return false;82 } else {83 jobFinished(params, false);84 return true;85 }86 }87 }

上面提到过JobService其实也是Service,同样具有相应生命周期方法。TestJobService先不做详细讲解,但是要看到这个类中封装了一些JobScheduler方法供外界调用。

接下来我们结合MainActivity类一起看,MainActivity源码如下:

1 public class MainActivity extends Activity {  2   3     private static final String TAG = "MainActivity";  4   5     public static final int MSG_UNCOLOUR_START = 0;  6     public static final int MSG_UNCOLOUR_STOP = 1;  7     public static final int MSG_SERVICE_OBJ = 2;  8     // UI fields.  9     int defaultColor; 10     int startJobColor; 11     int stopJobColor; 12  13     private TextView mShowStartView; 14     private TextView mShowStopView; 15     private TextView mParamsTextView; 16     private EditText mDelayEditText; 17     private EditText mDeadlineEditText; 18     private RadioButton mWiFiConnectivityRadioButton; 19     private RadioButton mAnyConnectivityRadioButton; 20     private CheckBox mRequiresChargingCheckBox; 21     private CheckBox mRequiresIdleCheckbox; 22     ComponentName mServiceComponent; 23     TestJobService mTestService; 24     // 25     private static int kJobId = 0; 26  27     @Override 28     public void onCreate(Bundle savedInstanceState) { 29         super.onCreate(savedInstanceState); 30         setContentView(R.layout.sample_main); 31         Resources res = getResources(); 32         defaultColor = res.getColor(R.color.none_received); 33         startJobColor = res.getColor(R.color.start_received); 34         stopJobColor = res.getColor(R.color.stop_received); 35         // Set up UI. 36         mShowStartView = (TextView) findViewById(R.id.onstart_textview); 37         mShowStopView = (TextView) findViewById(R.id.onstop_textview); 38         mParamsTextView = (TextView) findViewById(R.id.task_params); 39         mDelayEditText = (EditText) findViewById(R.id.delay_time); 40         mDeadlineEditText = (EditText) findViewById(R.id.deadline_time); 41         mWiFiConnectivityRadioButton = (RadioButton) findViewById(R.id.checkbox_unmetered); 42         mAnyConnectivityRadioButton = (RadioButton) findViewById(R.id.checkbox_any); 43         mRequiresChargingCheckBox = (CheckBox) findViewById(R.id.checkbox_charging); 44         mRequiresIdleCheckbox = (CheckBox) findViewById(R.id.checkbox_idle); 45         mServiceComponent = new ComponentName(this, TestJobService.class); 46         // Start service and provide it a way to communicate with us. 47         Intent startServiceIntent = new Intent(this, TestJobService.class); 48         startServiceIntent.putExtra("messenger", new Messenger(mHandler)); 49        startService(startServiceIntent); 50     } 51  52     Handler mHandler = new Handler(/* default looper */) { 53         @Override 54         public void handleMessage(Message msg) { 55             switch (msg.what) { 56                 case MSG_UNCOLOUR_START: 57                     mShowStartView.setBackgroundColor(defaultColor); 58                     break; 59                 case MSG_UNCOLOUR_STOP: 60                     mShowStopView.setBackgroundColor(defaultColor); 61                     break; 62                 case MSG_SERVICE_OBJ: 63                     mTestService = (TestJobService) msg.obj; 64                     mTestService.setUiCallback(MainActivity.this); 65             } 66         } 67     }; 68  69     private boolean ensureTestService() { 70         if (mTestService == null) { 71             Toast.makeText(MainActivity.this, "Service null, never got callback?", 72                     Toast.LENGTH_SHORT).show(); 73             return false; 74         } 75         return true; 76     } 77  78     /** 79      * UI onclick listener to schedule a job. What this job is is defined in 80      * TestJobService#scheduleJob(). 81      */ 82     @SuppressLint("NewApi") 83     public void scheduleJob(View v) { 84         if (!ensureTestService()) { 85             return; 86         } 87         JobInfo.Builder builder = new JobInfo.Builder(kJobId++, mServiceComponent); 88  89         String delay = mDelayEditText.getText().toString(); 90         if (delay != null && !TextUtils.isEmpty(delay)) { 91             builder.setMinimumLatency(Long.valueOf(delay) * 1000); 92         } 93         String deadline = mDeadlineEditText.getText().toString(); 94         if (deadline != null && !TextUtils.isEmpty(deadline)) { 95             builder.setOverrideDeadline(Long.valueOf(deadline) * 1000); 96         } 97         boolean requiresUnmetered = mWiFiConnectivityRadioButton.isChecked(); 98         boolean requiresAnyConnectivity = mAnyConnectivityRadioButton.isChecked(); 99         if (requiresUnmetered) {100             builder.setRequiredNetworkType(JobInfo.NETWORK_TYPE_UNMETERED);101         } else if (requiresAnyConnectivity) {102             builder.setRequiredNetworkType(JobInfo.NETWORK_TYPE_ANY);103         }104         builder.setRequiresDeviceIdle(mRequiresIdleCheckbox.isChecked());105         builder.setRequiresCharging(mRequiresChargingCheckBox.isChecked());106        mTestService.scheduleJob(builder.build());107     }108     /**109      * cancel All jobs110      * @param v111      */112     @SuppressLint("NewApi")113     public void cancelAllJobs(View v) {114         JobScheduler tm =115                 (JobScheduler) getSystemService(Context.JOB_SCHEDULER_SERVICE);116         tm.cancelAll();117     }118 119     /**120      * UI onclick listener to call jobFinished() in our service.121      */122     public void finishJob(View v) {123         if (!ensureTestService()) {124             return;125         }126         mTestService.callJobFinished();127         mParamsTextView.setText("");128     }129 130     /**131      * Receives callback from the service when a job has landed132      * on the app. Colours the UI and post a message to133      * uncolour it after a second.134      */135     @SuppressLint("NewApi")136     public void onReceivedStartJob(JobParameters params) {137         mShowStartView.setBackgroundColor(startJobColor);138         Message m = Message.obtain(mHandler, MSG_UNCOLOUR_START);139         mHandler.sendMessageDelayed(m, 1000L); // uncolour in 1 second.140         mParamsTextView.setText("Executing: " + params.getJobId() + " " + params.getExtras());141     }142 143     /**144      * Receives callback from the service when a job that145      * previously landed on the app must stop executing.146      * Colours the UI and post a message to uncolour it after a147      * second.148      */149     public void onReceivedStopJob() {150         mShowStopView.setBackgroundColor(stopJobColor);151         Message m = Message.obtain(mHandler, MSG_UNCOLOUR_STOP);152         mHandler.sendMessageDelayed(m, 2000L); // uncolour in 1 second.153         mParamsTextView.setText("");154     }155     156 }

对于这个Demo我只想说一个点,MainActivity在启动的时候onCreate()方法中47-49行启动了TestJobService,并且传递了一个new Messenger(mHandler)匿名对象,接下来我们看下TestJobService中onStartCommand方法,此方法上来就获取Messenger对象,并且调用send方法发送一个消息(实际调用的就是Handler的send方法,可自行点进源码查看),此消息obj为TestJobService类实例对象。回到MainActivity中查看mHandler,what为MSG_SERVICE_OBJ的时候:将msg的obj赋值给MainActivity中mTestService变量,然后调用TestJobService中setUiCallback方法,这样MainActivity与TestJobService类就互相持有彼此实例对象了,也就可以调用其内部的方法互相通信了,比如TestJobService中onStartJob方法调用MainActivity中的onReceivedStartJob方法,onReceivedStartJob方法中其实就是发送了Handler消息,最后在Handler中执行相关业务逻辑。好了Demo就说这一点,其余请自行细细查看,很简单,最后附上Demo布局文件:

1 
5 6
11 12
15
23
31
32
40 41
50 51
57
62
65
70
74
79
84
85 86
87
90
94
100
105
110
115
116
119
124
129
130
133
138
143
144 145 146
155
163 164

好了,本篇到此结束,希望通过本篇介绍你能正确使用JobScheduler这个强大的工具,有些任务放在适合的时机来做更加适宜。

 

转载于:https://www.cnblogs.com/leipDao/p/8268918.html

你可能感兴趣的文章
SD卡 TF卡 接口引脚定义
查看>>
STM32 逐次逼近寄存器型(SAR)模拟数字转换器(ADC)
查看>>
k8s认证及ServiceAccount-十五
查看>>
【数论】逆元详解
查看>>
精确软件延迟
查看>>
网络流24题 洛谷 2763 试题库问题
查看>>
解决:Unable to execute dex: GC overhead limit exceeded
查看>>
UICollectionViewCell 所遇到的问题
查看>>
flex使用学习
查看>>
Spring Cloud应用监控与管理Actuator
查看>>
H5 video的使用
查看>>
java提高篇-----字符串
查看>>
MFC中 使用Tab Control 控件在对话框中添加属性页
查看>>
asp网络编程:Web程序中网页间数据传递方法小结
查看>>
wust2012级软件工程新生经验交流会草稿
查看>>
CheeseZH: Stanford University: Machine Learning Ex2:Logistic Regression
查看>>
sql 查询所有子节点示例
查看>>
uva 1328(kmp)
查看>>
ES6
查看>>
浏览器前缀
查看>>