A Job Queue specifically written for Android to easily schedule jobs (tasks) that run in the background, improving UX and application stability.
Thanks to everybody who’ve used Android Priority JobQueue.
It was designed in a world where there was no JobScheduler, RxJava was not popular and Kotlin wasn’t even born publicly.
Today, most of the learnings in this project are part of WorkManager, the official deferred task library for Android. I’ve been involved in its development and I think it is the right way to do deferred tasks on Android.
For your persistent jobs, I recommend using WorkManager.
For your non-persistent jobs, drink the kool aid and use Coroutines.
Thanks.
There is a major internal rewrite of this project for more stability and new features. If you were using v1, see the migration guide here: migration from v1 to v2
dependencies {
compile 'com.birbit:android-priority-jobqueue:2.0.1'
}
Priority Job Queue is an implementation of a Job Queue specifically written for Android to easily schedule jobs (tasks) that run in the background, improving UX and application stability.
It is written primarily with flexibility & functionality in mind. This is an ongoing project, which we will continue to add stability and performance improvements.
Almost every application does work in a background thread. These “background tasks” are expected to keep the application responsive and robust, especially during unfavorable situations (e.g. limited network connectivity). In Android applications, there are several ways to implement background work:
Job Queue provides you a nice framework to do all of the above and more. You define your background tasks as Jobs and enqueue them to your JobManager instance. Job Manager will take care of prioritization, persistence, load balancing, delaying, network control, grouping etc. It also provides a nice lifecycle for your jobs to provide a better, consistent user experience.
Although not required, it is most useful when used with an event bus. It also supports dependency injection.
Since a code example is worth thousands of documentation pages, here it is.
File: PostTweetJob.java
// A job to send a tweet
public class PostTweetJob extends Job {
public static final int PRIORITY = 1;
private String text;
public PostTweetJob(String text) {
// This job requires network connectivity,
// and should be persisted in case the application exits before job is completed.
super(new Params(PRIORITY).requireNetwork().persist());
}
@Override
public void onAdded() {
// Job has been saved to disk.
// This is a good place to dispatch a UI event to indicate the job will eventually run.
// In this example, it would be good to update the UI with the newly posted tweet.
}
@Override
public void onRun() throws Throwable {
// Job logic goes here. In this example, the network call to post to Twitter is done here.
// All work done here should be synchronous, a job is removed from the queue once
// onRun() finishes.
webservice.postTweet(text);
}
@Override
protected RetryConstraint shouldReRunOnThrowable(Throwable throwable, int runCount,
int maxRunCount) {
// An error occurred in onRun.
// Return value determines whether this job should retry or cancel. You can further
// specify a backoff strategy or change the job's priority. You can also apply the
// delay to the whole group to preserve jobs' running order.
return RetryConstraint.createExponentialBackoff(runCount, 1000);
}
@Override
protected void onCancel(@CancelReason int cancelReason, @Nullable Throwable throwable) {
// Job has exceeded retry attempts or shouldReRunOnThrowable() has decided to cancel.
}
}
File: TweetActivity.java
//...
public void onSendClick() {
final String status = editText.getText().toString();
if(status.trim().length() > 0) {
jobManager.addJobInBackground(new PostTweetJob(status));
editText.setText("");
}
}
...
That’s it. 😃 Job Manager allows you to enjoy:
On Lollipop, Android introduced JobScheduler which is a system friendly way to run non-time-critical tasks. It makes your code cleaner, makes your app a good citizen of the ecosystem and it is backported via GCMNetworkManager.
The first version of Job Queue was created approximately 2 years before Job Scheduler. The major difference is that Job Queue is designed to run all of your background tasks while Job Scheduler is designed only for those you can defer.
We’ve created Job Queue because we wanted to have more control over the non-ui-thread activity of our application. We needed a convenient way to prioritize them, persist them accross application restarts and group based on the resources they access.
A good practice of using Job Queue is to write all of your network tasks as Jobs and use AsyncTasks for disk access (e.g. loading data from sqlite). If you have long running background operations (e.g. processing an image), it is also a good practice to use Job Queue.
Starting with v2, Job Queue can be integrated with JobScheduler or GCMNetworkManager. This integration allows Job Queue to wake up the aplication based on the criterias of the Jobs it has. You can see the details on the related wiki page. The Scheduler API is flexible such that you can implement a custom version of it if your target market does not have Google Play Services.
When user clicked the send button, onSendClick()
was called, which creates a PostTweetJob
and adds it to Job Queue for execution.
It runs on a background thread because Job Queue will make a disk access to persist the job.
Right after PostTweetJob
is synchronized to disk, Job Queue calls DependencyInjector
(if provided) which will inject fields into our job instance.
At PostTweetJob.onAdded()
callback, we saved PostTweetJob
to disk. Since there has been no network access up to this point, the time between clicking the send button and reaching onAdded()
is within fractions of a second. This allows the implementation of onAdded()
to inform UI to display the newly sent tweet almost instantly, creating a “fast” user experience.
In V1, onAdded()
is called on the thread job was added.
In V2, onAdded()
is called in JobManager’s own thread.
When it’s time for PostTweetJob
to run, Job Queue will call onRun()
(and it will only be called if there is an active network connection, as dictated at the job’s constructor).
By default, Job Queue uses a simple connection utility that checks ConnectivityManager
(ensure you have ACCESS_NETWORK_STATE
permission in your manifest). You can provide a custom implementation which can
add additional checks (e.g. your server stability). You should also provide a NetworkUtil
which can notify Job Queue when network
is recovered so that Job Queue will avoid a busy loop and decrease # of consumers(default configuration does it for you).
Job Queue will keep calling onRun()
until it succeeds (or reaches a retry limit). If onRun()
throws an exception,
Job Queue will call shouldReRunOnThrowable()
to allow you to handle the exception and decide whether to retry job execution or abort.
If all retry attempts fail (or when shouldReRunOnThrowable()
returns false), Job Queue will call onCancel()
to allow you to clean
your database, inform the user, etc.
AsyncTask
lifecycles. This is true assuming you use an event bus to update your UI (you should).SendMessageToNetwork
jobs, you can group them by conversation ID. Through this approach, messages in the same conversation will send in the order they were enqueued, while messages between different conversations are still sent in parallel. This lets you effortlessly maximize network utilization and ensure data integrity.NetworkUtil
if you need custom logic (e.g. you can create another instance of Job Queue which runs only if there is a wireless connection).We distribute artifacts through maven central repository.
Gradle: compile 'com.birbit:android-priority-jobqueue:3.0.0'
Maven:
<dependency>
<groupId>com.birbit</groupId>
<artifactId>android-priority-jobqueue</artifactId>
<version>3.0.0</version>
</dependency>
Gradle: compile 'com.birbit:android-priority-jobqueue:2.0.1'
Maven:
<dependency>
<groupId>com.birbit</groupId>
<artifactId>android-priority-jobqueue</artifactId>
<version>2.0.1</version>
</dependency>
You can also download library jar, sources and Javadoc from Maven Central.
We highly recommend checking how you can configure job manager and individual jobs.
Job#getApplicationContext
as a convenience method to get the Context inside a Job.BaseJob
in favor of a more complete Job
class.compile 'com.google.android.gms:play-services-gcm:10.0.1'
> cd jobqueue
> ./gradlew clean assembleDebug assembleDebugUnitTest test
This will create a jar file under release folder.
cd jobqueue
./gradlew clean check
Android Priority Jobqueue is made available under the MIT license:
The MIT License (MIT) Copyright (c) 2013 Path, Inc. Copyright (c) 2014 Google, Inc. Permission is hereby granted, free of charge, to any person obtaining a copy of this software and associated documentation files (the "Software"), to deal in the Software without restriction, including without limitation the rights to use, copy, modify, merge, publish, distribute, sublicense, and/or sell copies of the Software, and to permit persons to whom the Software is furnished to do so, subject to the following conditions: The above copyright notice and this permission notice shall be included in all copies or substantial portions of the Software. THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE SOFTWARE.