Target Audience
The target audience for this article is someone who has basic knowledge about asynchronous apex in Salesforce.
Perquisite
- Basic Knowledge about salesforce development concept
- Asynchronous apex in salesforce
Preface
In this article I will go through:
- Code Snippet for the given scenario
- Basic Terminology of batch apex
- Understanding code line by line
- Limits in batch Apex
- Interview Questions related to batch apex
Learning
- Understanding different interfaces in batch apex
- Start, finish, execute method in batch apex
- Getting success and failed record ids using database.stateful interface
- Run batch apex
- Understand singleEmailMessage class
- Scheduling batch job
Scenario
Update account description, number of employees, contact last name using batch apex. Get the failure record ids in the email. Also, schedule the job for every Monday.
1.Code Snippet
Running batch job
- Go to developer console
- Open execute anonymous window.
- Run the below code
Monitor status of batch job
To monitor the status of the jobs
- From Setup, enter Jobs in Quick Find box
- Select Apex Jobs
2.Understanding batch apex terminology
There is a governor limit on the total number of records retrieved by SOQL queries. We can only retrieve 50,000 records. But what happens if we want to retrieve more than 50,000 records. Here the batch apex comes as a savior. By using Batch apex class we can process the records in batches asynchronously. Each execution of a batch Apex job is considered a discrete transaction.
the basic structure of a batch consists of three methods:
- Start
- Execute
- Finish
Start method:
Use the start method to collect the records or objects to be passed to the interface method execute. This method execute only one time.
Execute method:
Execute method is called for each batch of records.
Batches of records are not guaranteed to execute in the order they are received from the start method.
Finish method:
The finish method is called after all batches are processed. Use this method to send confirmation emails or execute post-processing operations.
3.Understanding code line by line
Now that we have the understanding of basic terminology, its time to dig into the code.
Database.Batchable <sObject>
The class (batchUpdateAccount) implements Database. Batchable interface so that class can be executed as a batch Apex job. It has start, execute and finish method.
Database.Stateful
This interface is implemented to maintain state of the job across transaction. When using Database.Stateful, only instance member variables retain their values between transactions. Static member variables don’t retain their values and are reset between transactions.
If you don’t specify Database.Stateful, all static and instance member variables are set back to their original values.
Note: It’s important to know that using Database.Stateful will harm the batch’s performance, because the class will be serialized at the end of each execute method to update its internal state. This extra serialization results in longer execution time. There are alternative means to using Database.Stateful, such as storing data in a Custom Setting, which may offer better performance in some cases.
Schedulable Interface
This class implements Schedulable interface so that it can be scheduled to run at different intervals. It contains execute method for executing the apex job.
Start Method
Start Method below is collecting the account and related contact records to be passed to the interface method execute.
Database.QueryLocator
I am using here Database.QueryLocator. If you want to run batch on records that can be filtered by SOQL then QueryLocator is used.By using this the governor limit for the total number of records retrieved by SOQL queries is bypassed. (Default 50,000 It allow up to 50 million records).
Iteratable
Use iteratable if records that you want to be processed by batch can not be filtered by SOQL. If you use an iterable, the governor limit for the total number of records retrieved by SOQL queries is still enforced.
Update Account and Contact Records
List of Accounts retrieved from start method is passed to execute method to do the processing. Here I am iterating over account and contact records and then updating the records.
I have created list(accsToUpdate) to store updated account records and (cUpdate) to store updated contact records.
I created two sets to store success records and failed records respectively.
Database.Update
Use Database.Update method instead of update with allOrNothing = false parameter I am setting false parameter in Database.Update, it means that if one record is having exception then only that record will be failed to save, but any other records in the batch saves perfectly fine.
Database.SaveResult
The result of update DML operation returned by a Database method is saved in SaveResult class.
I iterate over the Database.SaveResult to get ID of successful and failed records.
Finish Method
As I have earlier stated that finish method in batch apex is used for post processing logic i.e. for sending email.
AsyncApexJob
Used AsyncApexJob is an object which contains various fields like status, NumberOfErrors, JobItemProcessed etc.. to get the information of current batch job
Below is the code to send email once the job is completed.
SingleEmailMesssage
SingleEmailMesssage is class, Messaging is name space.
setToAddress(toAddress)
List of String to store email address for the email to sent and then use method setToAddress() to send email.
setSubject(Subject)
Email subject line is sent by setSubject method
setPlainTextBody(plainTextBody)
This method is used to send the text of the email. I am mailing successRecords, failure records,number of errors, totalJobitems in the email text
sendEmail(emails, allOrNothing)
Sends the list of up to 10 emails instantiated with either SingleEmailMessage or MassEmailMessage and returns a list of SendEmailResult objects.
new Messaging.SingleEmailMessage[] { mail } is the list of email messages, they create it here on the fly
The [] means it is a list
Messaging.sendEmail(new Messaging.SingleEmailMessage[] { mail1, mail2, mail3 } );
Email received once batch job is run
Schedule Apex
- From Setup, enter Apex Class in Quick Find box
- Select Schedule Apex
Once you select schedule apex, you will see a below screen.
To see all scheduled jobs
From Setup, enter Scheduled jobs in Quick Find box
The moment you schedule the apex class, the job gets queued.
Note: Till the time job is not completed, you cannot make changes to the apex class (batchUpdateAccountsContacts)
4.Limits in batch Apex
- Up to five queued or active batch jobs are allowed for Apex
- A maximum of 50 million records can be returned to the Database.QueryLocator object.
- If the start method returns a QueryLocator, the optional scope parameter of Database.executeBatch can have a maximum value of 2,000. If set to a higher value, Salesforce chunks the records returned by the QueryLocator into smaller batches of up to 2,000 records.
- If the start method returns an iterable, the scope parameter value has no upper limit; however, if you use a very high number, you may run into other limits.
- If no size is specified with the optional scope parameter of Database.executeBatch, Salesforce chunks the records returned by the start method into batches of 200, and then passes each batch to the execute method.
- The start, execute, and finish methods can implement up to 10 callouts each
- The maximum number of batch executions is 250,000 per 24 hours
- Only one batch Apex job’s start method can run at a time in an organization. Batch jobs that haven’t started yet remain in the queue until they’re started.
5.Interview Questions
What if you change the name of Execute method to Execute1 in the batch class? Will the batch job still run?
Go ahead and change Execute to Excute1 and try saving the class.
Output
Class batchUpdateAccountsContacts must implement the method: void Database.Batchable<SObject>.execute(Database.BatchableContext, List<SObject>)
Finding
It won let you save the batch class as it says class must implement execute method
Can we call a batch class from another batch class?
Yes, we can call one batch class from another batch class but only if it is called from its finish method.
Can we call a Future method from batch class?
No, we cannot directly call a future method from a batch class.
Is there is any way through which we can call future method from batch apex?
As we know that a webservice can be called from batch class and webservice can call @future method. So in your batch class call web service and which can call your @future method. Also, you can call future methods from the finish method in the batch class.
How many can concurrent batch jobs be added to the queue?
At most, there can be 5 bath jobs queued at a time.
How can we track the status of the current running batch job?
The job Id returned by the batchable context variable helps us in finding the status of a batch through AsyncApexJob.Status.
Is it possible to do Synchronous Web service callouts from scheduled apex?
No.Synchronous Web service callouts are not supported by scheduled Apex. However, if your scheduled Apex executes a batch job, callouts are supported by the batch class.
Can we call the scheduler from the future method?
Yes
Can we call the scheduler from the future method?
Yes
Can we modify the scheduler class or classes referenced by this scheduler class if any scheduled job is pending?
No. If there are one or more active scheduled jobs for an Apex class, you cannot update the class or any classes referenced by this class through the Salesforce user interface.
Hopefully, this scenario will help in clearly understanding batch apex. For more of these kinds of articles stay tuned. Happy Coding!
No comments:
Post a Comment