Skip to main content

Control Processes with Queueable Apex

Learning Objectives

After completing this unit, you’ll know:

  • When to use the Queueable interface.
  • The differences between queueable and future methods.
  • Queueable Apex syntax.
  • Queueable method best practices.

Follow Along with Trail Together

Want to follow along with an instructor as you work through this step? Take a look at this video, which is part of the Trail Together series on Trailhead Live.

(This clip starts at the 52:38 minute mark, in case you want to rewind and watch the beginning of the step again.)

Queueable Apex

Queueable Apex is essentially a superset of future methods with these extra benefits:

  • Non-primitive types: Your Queueable class can contain member variables of non-primitive data types, such as sObjects or custom Apex types. Those objects can be accessed when the job executes.
  • Monitoring: When you submit your job by invoking the System.enqueueJob() method, the method returns the ID of the AsyncApexJob record. You can use this ID to identify your job and monitor its progress, either through the Salesforce user interface in the Apex Jobs page, or programmatically by querying your record from AsyncApexJob.
  • Chaining jobs: You can chain one job to another job by starting a second job from a running job. Chaining jobs is useful if you need to do some sequential processing.

Queueable Versus Future

Because queueable methods are functionally equivalent to future methods, most of the time you’ll probably want to use queueable instead of future methods. However, this doesn’t necessarily mean you should go back and refactor all your future methods right now.

Another reason to use future methods instead of queueable Apex is when your functionality is sometimes executed synchronously, and sometimes asynchronously. It’s much easier to refactor a method in this manner than converting to a queueable class. This is handy when you discover that part of your existing code needs to be moved to async execution. You can simply create a similar future method that wraps your synchronous method like so:

@future
static void myFutureMethod(List<String> params) {
    // call synchronous method
    mySyncMethod(params);
}

Queueable Syntax

To use queueable Apex, simply implement the Queueable interface.

public class SomeClass implements Queueable {
    public void execute(QueueableContext context) {
        // awesome code here
    }
}

Sample Code

A common queuable Apex workflow is to get a set of sObject records, process them, and then update them in the database asynchronously. Because future methods are limited to primitive data types (or collections of primitives), queueable Apex is an ideal choice. The following code takes a collection of account records, sets the parentId for each record, and then updates the records in the database.

public class UpdateParentAccount implements Queueable {
    private List<Account> accounts;
    private ID parent;
    public UpdateParentAccount(List<Account> records, ID id) {
        this.accounts = records;
        this.parent = id;
    }
    public void execute(QueueableContext context) {
        for (Account account : accounts) {
          account.parentId = parent;
          // perform other processing or callout
        }
        update accounts;
    }
}

To add this class as a job on the queue, execute the following code:

// find all accounts in 'NY'
List<Account> accounts = [SELECT Id FROM Account WHERE BillingState = 'NY'];
// find a specific parent account for all records
Id parentId = [SELECT Id FROM Account WHERE Name = 'ACME Corp'][0].Id;
// instantiate a new instance of the Queueable class
UpdateParentAccount updateJob = new UpdateParentAccount(accounts, parentId);
// enqueue the job for processing
ID jobID = System.enqueueJob(updateJob);

After you submit your queueable class for execution, the job is added to the queue and will be processed when system resources become available.

You can use the new job ID to monitor progress, either through the Apex Jobs page or programmatically by querying AsyncApexJob:

SELECT Id, Status, NumberOfErrors FROM AsyncApexJob WHERE Id = :jobID

Testing Queueable Apex

The following code sample shows how to test the execution of a queueable job in a test method. It looks very similar to batch Apex testing.

The setup() method is annotated with @testSetup. This annotation allows you to create common test records that are available for all the methods in the class. Any updates to the test records are rolled back after each test method finishes executing. Although this test class includes only one method (besides the setup() method itself), we included the @testSetup annotation here as an example. In this case, the setup() method creates a parent account record and 100 child account records. It then inserts the records into the database. This data is used by the queueable class.

To ensure that the queueable process runs within the test method, the job is submitted to the queue between the Test.startTest and Test.stopTest block. The system executes all asynchronous processes started in a test method synchronously after the Test.stopTest statement. Next, the test method verifies the results of the queueable job by querying the account records that the job updated.

@isTest
public class UpdateParentAccountTest {
    @testSetup
    static void setup() {
        List<Account> accounts = new List<Account>();
        // add a parent account
        accounts.add(new Account(name='Parent'));
        // add 100 child accounts
        for (Integer i = 0; i < 100; i++) {
            accounts.add(new Account(
                name='Test Account'+i
            ));
        }
        insert accounts;
    }
    @isTest
    static void testQueueable() {
        // query for test data to pass to queueable class
        Id parentId = [SELECT Id FROM Account WHERE Name = 'Parent'][0].Id;
        List<Account> accounts = [SELECT Id, Name FROM Account WHERE Name LIKE 'Test Account%'];
        // Create our queueable instance
        UpdateParentAccount updater = new UpdateParentAccount(accounts, parentId);
        // startTest/stopTest block to force async processes to run
        Test.startTest();
        System.enqueueJob(updater);
        Test.stopTest();
        // Validate the job ran. Check if record have correct parentId now
        System.assertEquals(100, [SELECT count() FROM Account WHERE parentId = :parentId]);
    }
}



Chaining Jobs

One of the best features of queueable Apex is job chaining. If you ever need to run jobs sequentially, queueable Apex can make your life much easier. To chain a job to another job, submit the second job from the execute() method of your queueable class. You can add only one job from an executing job, which means that only one child job can exist for each parent job. For example, if you have a second class called SecondJob that implements the Queueable interface, you can add this class to the queue in the execute() method as follows:

public class FirstJob implements Queueable {
    public void execute(QueueableContext context) {
        // Awesome processing logic here
        // Chain this job to next job by submitting the next job
        System.enqueueJob(new SecondJob());
    }
}

You can use defined stack depths to test chains of queueable jobs, but be aware of applicable Apex governor limits. See Adding a Queueable Job with a Specified Stack Depth.

Things to Remember

Queueable Apex is a great tool but there are a few things to watch out for:

  • The execution of a queued job counts once against the shared limit for asynchronous Apex method executions.
  • You can add up to 50 jobs to the queue with System.enqueueJob in a single transaction.
  • When chaining jobs, you can add only one job from an executing job with System.enqueueJob, which means that only one child job can exist for each parent queueable job. You can’t start multiple child jobs from the same queueable job.
  • No limit is enforced on the depth of chained jobs, which means that you can chain one job to another job and repeat this process with each new child job. However, for Developer Edition and Trial orgs, the maximum stack depth for chained jobs is 5, which means that you can chain jobs four times, and the maximum number of jobs in the chain is 5, including the initial parent queueable job. However, you can override this limit by using the Configure Stack Depth of Chained Queueable Jobs feature.

Resources

Condividi il tuo feedback su Trailhead dalla Guida di Salesforce.

Conoscere la tua esperienza su Trailhead è importante per noi. Ora puoi accedere al modulo per l'invio di feedback in qualsiasi momento dal sito della Guida di Salesforce.

Scopri di più Continua a condividere il tuo feedback