JobService和JobScheduler机制在Android5.0以上保活

我们知道在Android5.0之前,Android源代码还是有不小漏洞的,导致非常多不光明的手段来进行++保活++。但是在Android5.0之后。非常多都是能够被APP杀死的。Android5.0之后Android提供了JobService和JobScheduler这两的类。我们能够通过这个JobScheduler来进行保活。

JobScheduler

JobScheduler是Job的调度类。负责运行。取消任务等逻辑。详细看下JobScheduler的获取和类代码。

 JobScheduler jobScheduler = (JobScheduler)getSystemService(Context.JOB_SCHEDULER_SERVICE)
/*
*參数:JobInfo採用Builder的设计模式,对须要运行的Job任务信息进行的封装。
*返回值:RESULT_SUCCESS=1 RESULT_FAILURE=0 表示运行成功或失败
*/
public abstract int schedule(JobInfo job); /**通过指定的jobId取消Job任务*/
public abstract void cancel(int jobId); /**取消全部的Job任务*/
public abstract void cancelAll(); /**获取全部的未运行的Job任务*/
public abstract @NonNull List<JobInfo> getAllPendingJobs(); /**获取指定的Job未运行的任务*/
public abstract @Nullable JobInfo getPendingJob(int jobId);

JobService

JobService中的代码实现不多,内部使用AIDL + Handler的方式来传递消息,当中重要的就是这几个方法。

/*
*须要重写,開始jobScheduler的方法
*/
public abstract boolean onStartJob(JobParameters params); /*
*停止JobScheduler的方法
*/
public abstract boolean onStopJob(JobParameters params); /*
*完毕JobScheduler的方法
*/
public final void jobFinished(JobParameters params, boolean needsReschedule) {
ensureHandler();
Message m = Message.obtain(mHandler, MSG_JOB_FINISHED, params);
m.arg2 = needsReschedule ? 1 : 0;
m.sendToTarget();
}

在JobService中,这几个方法都是通过Handler发送Message。在Handler中调用了IJobCallback的底层的实现。

JobInfo

// jobId每一个Job任务的id
int jobId = 1;
// 指定你须要运行的JobService
ComponentName name = new ComponentName(getPackageName(), MyJobService.class.getName())); JobInfo.Builder builder = new JobInfo.Bulider(jobId, name); builder.setRequiredNetworkType(JobInfo.NETWORK_TYPE_NONE); //设置须要的网络条件,默认NETWORK_TYPE_NONE builder.setPeriodic(3000);//设置间隔时间 builder.setMinimumLatency(3000);// 设置任务运行最少延迟时间 builder.setOverrideDeadline(50000);// 设置deadline,若到期还没有达到规定的条件则会開始运行 builder.setRequiresCharging(true);// 设置是否充电的条件,默认false builder.setRequiresDeviceIdle(false);// 设置手机是否空暇的条件,默认false builder.setPersisted(true);//设备重新启动之后你的任务是否还要继续运行 JobInfo info = builder.build();

演示样例代码实现

在MainActivity中运行JobScheduler的scheduler()方法

public class MainActivity extends AppCompatActivity {

    @Override
protected void onCreate(Bundle savedInstanceState) {
super.onCreate(savedInstanceState);
setContentView(R.layout.activity_main); Button button = (Button) findViewById(R.id.btn);
button.setText(getClass().getSimpleName());
button.setOnClickListener(new View.OnClickListener() {
@Override
public void onClick(View v) {
if (Build.VERSION.SDK_INT >= Build.VERSION_CODES.LOLLIPOP) {
JobScheduler jobScheduler = (JobScheduler) getSystemService(JOB_SCHEDULER_SERVICE);
JobInfo jobInfo = new JobInfo.Builder(1, new ComponentName(getPackageName(), MyJobService.class.getName()))
.setPeriodic(2000)
.setRequiredNetworkType(JobInfo.NETWORK_TYPE_ANY)
.build();
jobScheduler.schedule(jobInfo);
}
}
});
}
}

MyService类

@RequiresApi(api = Build.VERSION_CODES.LOLLIPOP)//API须要在21及以上
public class MyJobService extends JobService { private Handler handler = new Handler(new Handler.Callback() {
@Override
public boolean handleMessage(Message msg) {
Toast.makeText(MyJobService.this, "MyJobService", Toast.LENGTH_SHORT).show();
JobParameters param = (JobParameters) msg.obj;
jobFinished(param, true);
Intent intent = new Intent(getApplicationContext(), MainActivity.class);
intent.addFlags(Intent.FLAG_ACTIVITY_NEW_TASK);
startActivity(intent);
return true;
}
}); @Override
public int onStartCommand(Intent intent, int flags, int startId) {
return START_STICKY;
} @Override
public boolean onStartJob(JobParameters params) {
Message m = Message.obtain();
m.obj = params;
handler.sendMessage(m);
return true;
} @Override
public boolean onStopJob(JobParameters params) {
handler.removeCallbacksAndMessages(null);
return false;
}
}

Mainfiest.xml中注冊MyServcie

    <service
android:name=".MyJobService"
android:permission="android.permission.BIND_JOB_SERVICE">

在大多数Android5.0以上手机杀死APP进程后,仍然能够唤醒,手机重新启动在部分手机但是能够唤醒APP的。

JobScheduler还有待深入。这里推荐几篇比較好的博客。

Android之JobScheduler运行机制源代码分析

在Android 5.0中使用JobScheduler

5
0
05-11 17:34