Apex - Stapelverarbeitung
In diesem Kapitel werden wir die Stapelverarbeitung in Apex verstehen. Stellen Sie sich ein Szenario vor, in dem wir täglich eine große Anzahl von Datensätzen verarbeiten, wahrscheinlich die Bereinigung von Daten oder das Löschen nicht verwendeter Daten.
Was ist Batch Apex?
Batch Apex ist eine asynchrone Ausführung von Apex-Code, der speziell für die Verarbeitung der großen Anzahl von Datensätzen entwickelt wurde und eine größere Flexibilität bei den Governor-Limits aufweist als der synchrone Code.
Wann soll Batch Apex verwendet werden?
Wenn Sie eine große Anzahl von Datensätzen täglich oder sogar zu einem bestimmten Zeitpunkt verarbeiten möchten, können Sie sich für Batch Apex entscheiden.
Wenn eine Operation asynchron sein soll, können Sie auch den Batch-Apex implementieren. Batch Apex wird als Schnittstelle verfügbar gemacht, die vom Entwickler implementiert werden muss. Stapeljobs können zur Laufzeit mithilfe von Apex programmgesteuert aufgerufen werden. Batch Apex verarbeitet kleine Stapel von Datensätzen, deckt Ihren gesamten Datensatz ab und unterteilt die Verarbeitung in überschaubare Datenblöcke.
Verwenden von Batch Apex
Wenn wir den Batch-Apex verwenden, müssen wir die von Salesforce bereitgestellte Schnittstelle Database.Batchable implementieren und dann die Klasse programmgesteuert aufrufen.
Sie können die Klasse folgendermaßen überwachen:
Um die Ausführung des Apex-Batch-Batch-Jobs zu überwachen oder zu stoppen, gehen Sie zu Setup → Überwachung → Apex-Jobs oder Jobs → Apex-Jobs.
Die Database.Batchable-Schnittstelle verfügt über die folgenden drei Methoden, die implementiert werden müssen:
- Start
- Execute
- Finish
Lassen Sie uns nun jede Methode im Detail verstehen.
Start
Die Start-Methode ist eine der drei Methoden der Database.Batchable-Schnittstelle.
Syntax
global void execute(Database.BatchableContext BC, list<sobject<) {}
Diese Methode wird zu Beginn des Stapeljobs aufgerufen und sammelt die Daten, mit denen der Stapeljob ausgeführt wird.
Berücksichtigen Sie die folgenden Punkte, um die Methode zu verstehen:
Verwenden Sie die Database.QueryLocatorObjekt, wenn Sie eine einfache Abfrage verwenden, um den Umfang der im Stapeljob verwendeten Objekte zu generieren. In diesem Fall wird das SOQL-Datenzeilenlimit umgangen.
Verwenden Sie das iterierbare Objekt, wenn Sie komplexe Kriterien zum Verarbeiten der Datensätze haben. Database.QueryLocator bestimmt den Umfang der Datensätze, die verarbeitet werden sollen.
Ausführen
Lassen Sie uns nun die Execute-Methode der Database.Batchable-Schnittstelle verstehen.
Syntax
global void execute(Database.BatchableContext BC, list<sobject<) {}
Dabei wird list <sObject <von der Database.QueryLocator-Methode zurückgegeben.
Diese Methode wird nach der Start-Methode aufgerufen und führt die gesamte für den Stapeljob erforderliche Verarbeitung durch.
Fertig
Wir werden nun die Finish-Methode der Database.Batchable-Schnittstelle diskutieren.
Syntax
global void finish(Database.BatchableContext BC) {}
Diese Methode wird am Ende aufgerufen und Sie können einige Abschlussaktivitäten ausführen, z. B. das Senden einer E-Mail mit Informationen zu den verarbeiteten Stapeljobdatensätzen und zum Status.
Batch-Apex-Beispiel
Betrachten wir ein Beispiel unseres bestehenden Chemieunternehmens und gehen wir davon aus, dass wir das Feld Kundenstatus und Kundenbeschreibung der Kundendatensätze aktualisieren müssen, die als aktiv markiert wurden und das Datum wie heute erstellt haben. Dies sollte täglich erfolgen und eine E-Mail an einen Benutzer über den Status der Stapelverarbeitung gesendet werden. Aktualisieren Sie den Kundenstatus als "Verarbeitet" und die Kundenbeschreibung als "Über Stapeljob aktualisiert".
// Batch Job for Processing the Records
global class CustomerProessingBatch implements Database.Batchable<sobject> {
global String [] email = new String[] {'[email protected]'};
// Add here your email address here
// Start Method
global Database.Querylocator start (Database.BatchableContext BC) {
return Database.getQueryLocator('Select id, Name, APEX_Customer_Status__c,
APEX_Customer_Decscription__c From APEX_Customer__c WHERE createdDate = today
AND APEX_Active__c = true');
// Query which will be determine the scope of Records fetching the same
}
// Execute method
global void execute (Database.BatchableContext BC, List<sobject> scope) {
List<apex_customer__c> customerList = new List<apex_customer__c>();
List<apex_customer__c> updtaedCustomerList = new List<apex_customer__c>();
// List to hold updated customer
for (sObject objScope: scope) {
APEX_Customer__c newObjScope = (APEX_Customer__c)objScope ;
// type casting from generic sOject to APEX_Customer__c
newObjScope.APEX_Customer_Decscription__c = 'Updated Via Batch Job';
newObjScope.APEX_Customer_Status__c = 'Processed';
updtaedCustomerList.add(newObjScope); // Add records to the List
System.debug('Value of UpdatedCustomerList '+updtaedCustomerList);
}
if (updtaedCustomerList != null && updtaedCustomerList.size()>0) {
// Check if List is empty or not
Database.update(updtaedCustomerList); System.debug('List Size '
+ updtaedCustomerList.size());
// Update the Records
}
}
// Finish Method
global void finish(Database.BatchableContext BC) {
Messaging.SingleEmailMessage mail = new Messaging.SingleEmailMessage();
// Below code will fetch the job Id
AsyncApexJob a = [Select a.TotalJobItems, a.Status, a.NumberOfErrors,
a.JobType, a.JobItemsProcessed, a.ExtendedStatus, a.CreatedById,
a.CompletedDate From AsyncApexJob a WHERE id = :BC.getJobId()];
// get the job Id
System.debug('$$$ Jobid is'+BC.getJobId());
// below code will send an email to User about the status
mail.setToAddresses(email);
mail.setReplyTo('[email protected]'); // Add here your email address
mail.setSenderDisplayName('Apex Batch Processing Module');
mail.setSubject('Batch Processing '+a.Status);
mail.setPlainTextBody('The Batch Apex job processed'
+ a.TotalJobItems+'batches with '+a.NumberOfErrors+'failures'+'Job Item
processed are'+a.JobItemsProcessed);
Messaging.sendEmail(new Messaging.Singleemailmessage [] {mail});
}
}
Um diesen Code auszuführen, speichern Sie ihn zuerst und fügen Sie dann den folgenden Code in Anonym ausführen ein. Dadurch wird das Objekt der Klasse erstellt und die Methode Database.execute führt den Stapeljob aus. Sobald der Auftrag abgeschlossen ist, wird eine E-Mail an die angegebene E-Mail-Adresse gesendet. Stellen Sie sicher, dass Sie einen Kundendatensatz haben, der hatActive wie geprüft.
// Paste in Developer Console
CustomerProessingBatch objClass = new CustomerProessingBatch();
Database.executeBatch (objClass);
Überprüfen Sie nach Ausführung dieser Klasse die von Ihnen angegebene E-Mail-Adresse, an der Sie die E-Mail mit Informationen erhalten. Sie können den Status des Stapeljobs auch über die Seite Überwachung und die oben angegebenen Schritte überprüfen.
Wenn Sie die Debug-Protokolle überprüfen, finden Sie die Listengröße, die angibt, wie viele Datensätze verarbeitet wurden.
Limitations
Wir können jeweils nur 5 Stapeljobs bearbeiten. Dies ist eine der Einschränkungen von Batch Apex.
Planen des Apex-Stapeljobs mithilfe der Apex-Detailseite
Sie können die Apex-Klasse über die unten angegebene Apex-Detailseite planen.
Step 1 - Gehen Sie zu Setup ⇒ Apex-Klassen, klicken Sie auf Apex-Klassen.
Step 2 - Klicken Sie auf die Schaltfläche Schedule Apex.
Step 3 - Geben Sie Details an.
Planen des Apex-Stapeljobs über die planbare Schnittstelle
Sie können den Apex-Stapeljob über die planbare Schnittstelle wie folgt planen:
// Batch Job for Processing the Records
global class CustomerProessingBatch implements Database.Batchable<sobject> {
global String [] email = new String[] {'[email protected]'};
// Add here your email address here
// Start Method
global Database.Querylocator start (Database.BatchableContext BC) {
return Database.getQueryLocator('Select id, Name, APEX_Customer_Status__c,
APEX_Customer_Decscription__c From APEX_Customer__c WHERE createdDate = today
AND APEX_Active__c = true');
// Query which will be determine the scope of Records fetching the same
}
// Execute method
global void execute (Database.BatchableContext BC, List<sobject> scope) {
List<apex_customer__c> customerList = new List<apex_customer__c>();
List<apex_customer__c> updtaedCustomerList = new
List<apex_customer__c>();//List to hold updated customer
for (sObject objScope: scope) {
APEX_Customer__c newObjScope = (APEX_Customer__c)objScope ;//type
casting from generic sOject to APEX_Customer__c
newObjScope.APEX_Customer_Decscription__c = 'Updated Via Batch Job';
newObjScope.APEX_Customer_Status__c = 'Processed';
updtaedCustomerList.add(newObjScope);//Add records to the List
System.debug('Value of UpdatedCustomerList '+updtaedCustomerList);
}
if (updtaedCustomerList != null && updtaedCustomerList.size()>0) {
// Check if List is empty or not
Database.update(updtaedCustomerList); System.debug('List Size'
+ updtaedCustomerList.size());
// Update the Records
}
}
// Finish Method
global void finish(Database.BatchableContext BC) {
Messaging.SingleEmailMessage mail = new Messaging.SingleEmailMessage();
// Below code will fetch the job Id
AsyncApexJob a = [Select a.TotalJobItems, a.Status, a.NumberOfErrors,
a.JobType, a.JobItemsProcessed, a.ExtendedStatus, a.CreatedById,
a.CompletedDate From AsyncApexJob a WHERE id = :BC.getJobId()];//get the job Id
System.debug('$$$ Jobid is'+BC.getJobId());
// below code will send an email to User about the status
mail.setToAddresses(email);
mail.setReplyTo('[email protected]');//Add here your email address
mail.setSenderDisplayName('Apex Batch Processing Module');
mail.setSubject('Batch Processing '+a.Status);
mail.setPlainTextBody('The Batch Apex job processed'
+ a.TotalJobItems+'batches with '+a.NumberOfErrors+'failures'+'Job Item
processed are'+a.JobItemsProcessed);
Messaging.sendEmail(new Messaging.Singleemailmessage [] {mail});
}
// Scheduler Method to scedule the class
global void execute(SchedulableContext sc) {
CustomerProessingBatch conInstance = new CustomerProessingBatch();
database.executebatch(conInstance,100);
}
}
// Paste in Developer Console
CustomerProessingBatch objClass = new CustomerProcessingBatch();
Database.executeBatch (objClass);