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);